Swift 6.0 beta 日本語化計画 : Swift 6.0 beta
プロパティ
プロパティ は、特定のクラス、構造体、または列挙型と値を関連付けます。計算されたプロパティが (むしろ格納よりも) 値を計算するのに対して、格納されたプロパティは、インスタンスの一部として定数と変数の値を格納します。計算されたプロパティは、クラス、構造体、列挙型によって提供されます。格納されたプロパティはクラスおよび構造体によってのみ提供されます。
格納されたプロパティと計算されたプロパティは、通常、特定の型のインスタンスと関連しています。しかし、プロパティはまた、型自体に関連することもできます。このようなプロパティは、型プロパティとして知られています。
加えて、カスタム・アクションと共に応答できるプロパティの値の変化を監視するプロパティ監視者を定義できます。プロパティ監視者は、あなた自身が定義する格納されたプロパティに追加することができ、サブクラスがそのスーパークラスから継承するプロパティにも、追加できます。
プロパティラッパーを使用して、複数のプロパティのゲッタとセッタでコードを再利用することもできます。
その最も単純な形式では、格納されたプロパティは、特定のクラスまたは構造体のインスタンスの一部として格納されている定数または変数です。格納されたプロパティは、(var キーワードに引き続き導入された) 変数に格納されたプロパティ または (let キーワードに引き続き導入された) 定数に格納されたプロパティ のいずれかになります。
デフォルトのプロパティ値 で説明したように、その定義の一部として格納されたプロパティのデフォルト値を提供できます。また、初期化中に、格納されたプロパティの初期値を設定し、変更する事もできます。初期化中に定数プロパティへの代入 で説明したように、これは定数に格納されたプロパティについても正しいです。
以下の例では、FixedLengthRange という構造体を定義し、一旦作成されたら変更できない範囲の長さの整数の範囲を記述しています。
FixedLengthRange のインスタンスには、firstValue という変数に格納されたプロパティと、length という定数に格納されたプロパティがあります。上記の例では、length は新しい範囲が作成された時に初期化され、その後は、それが定数プロパティのため変更できません。
構造体のインスタンスを作成し、定数にそのインスタンスを代入した場合、それらが変数プロパティとして宣言された場合でも、インスタンス・プロパティを変更することはできません。
rangeOfFourItems は (let キーワードを使って) 定数として宣言されているので、firstValue が変数プロパティであっても、その firstValue プロパティを変更することはできません。
この動作は、値型 である構造体によるものです。値型のインスタンスが定数としてマークされると、そのすべてのプロパティが定数になります。
同じことは 参照型 であるクラスの場合、正しくありません。参照型のインスタンスを定数に代入した場合、まだそのインスタンスの変数プロパティを変更することができます。
遅延した格納されたプロパティ は、初めて使用されるまではその初期値が計算されないプロパティです。その宣言の前に lazy の修飾子を書くことによって、遅延した格納されたプロパティを示して下さい。
遅延したプロパティは、プロパティの初期値が、インスタンスの初期化が完了するまで値がわからない外部ファクタに依存する場合に便利です。遅延したプロパティはまた、プロパティの初期値が複雑であるか計算的に高価なセットアップを必要とする場合にも有用で、それが必要とされない限り、実行されません。
以下の例は、複雑なクラスで不必要な初期化を避けるために、遅延した格納されたプロパティを使用しています。この例では、2つとも完全には示されていない DataImporter と DataManager と言う2つのクラスを定義しています。
DataManager クラスには、String 値の新しい、空の配列で初期化された data と言う格納されたプロパティがあります。その機能の残りの部分は示されていませんが、この DataManagaer クラスの目的は String データのこの配列へのアクセスを管理し、提供することにあります。
DataManager クラスの機能の一部は、ファイルからデータをインポートする機能です。この機能は、 DataImporter クラスによって提供され、初期化するためには少なくない時間が掛かると仮定されます。これは、 DataImporter インスタンスがファイルを開いて、DataImporter インスタンスが初期化されたときにその内容をメモリに読み込む必要があるからです。
DataManager インスタンスは、ファイルからデータをインポートすることが全くなく、そのデータを管理することが可能であるため、DataManager 自体が作成されても、新しいDataImporter インスタンスを作成しません。その代わりに、それを最初に使用する際 DataImporter インスタンスを作成するのが、より理にかなっています。
lazy 修飾子でマークされているため、その fileName プロパティが照会されたときのように、importer プロパティが最初にアクセスされたとき importer プロパティの DataImporter インスタンスのみが作成されます。
Objective-C の経験がある場合、それがクラスインスタンスの一部として値と参照を格納するために 2つ の方法を提供することを知っているかもしれません。プロパティに加えて、プロパティに格納された値の補助記憶としてインスタンス変数を使用できます。
Swift は、これらの概念を一つのプロパティ宣言に統合します。Swift のプロパティは、対応するインスタンス変数を持っておらず、プロパティの補助記憶には直接アクセスできません。このアプローチは、どう値が異なる文脈でアクセスされたかについての混乱を回避し、一つの定義文にプロパティの宣言を簡素化します。プロパティに関するすべての情報は、その名前、型、およびメモリ管理特性を含め、型の定義の一部として一つの場所で定義されます。
格納されたプロパティ、クラス、構造体、および列挙型に加えて、実際には値を格納しない、計算されたプロパティ を定義できます。代わりに、それらは間接的に他のプロパティと値を取得し設定するためのゲッタと optional のセッタを提供します。
この例では、幾何学的形状を操作するため3つの構造体を定義しています。
Rect 構造体はまた、center と言う計算されたプロパティも提供します。Rect の現在の中心位置は、常にその origin と size から決定できるので、明示的な Point 値として中心点を格納する必要はありません。その代わり、Rect は、それが本当の格納されたプロパティであるかのように長方形の center で作業できるよう、center と言う計算された変数のカスタムゲッタとセッタを定義します。
上述の例では、square と言う新しい Rect 変数を作成しています。square 変数は (0,0) の原点で初期化され、幅及び高さは 10 です。この正方形は、下の図の緑の正方形で表されています。
square 変数の center プロパティは、現在のプロパティ値を取得するために、ドット構文 (square.center) を介してアクセスされ、center のゲッタの呼び出しを引き起こします。既存の値を返すよりもむしろ、ゲッタは、実際に計算し、正方形の中心を表すために新しい Point を返します。上記から分かるように、ゲッタは正しく (5,5) の中心点を返します。
center プロパティは、下の図では深緑色の正方形で示される新しい位置に、正方形を上、右にに移動し、(15、15) の新しい値に設定されます。center プロパティを設定すると、格納された origin プロパティの x と y の値を変更する center 用セッタを呼び出し、その新しい位置に正方形を移動します。
計算されたプロパティのセッタが、設定すべき新しい値の名前を定義していなければ、デフォルトの名前である newValue が使用されます。ここで、この省略形の表記法を利用している、Rect 構造体の別バージョンを示します:
ゲッタ (getter) の本体全体が単一の式である場合、ゲッタはその式を暗黙的に返します。この省略表記とセッタ (setter) の省略表記を利用する Rect 構造体の別のバージョンを次に示します。
ゲッタからの return を省略することは、暗黙の戻り値を持つ関数 で説明したように、関数からの return を省略することと同じ規則に従います。
ゲッタと計算されたプロパティがあり、セッタがないのは 読み取り専用の計算されたプロパティ として知られています。読み取り専用の計算されたプロパティは常に値を返し、ドット構文を介してアクセスできますが、別の値に設定することはできません。
get キーワードとその括弧を除くと、読み取り専用の計算されたプロパティの宣言を単純化できます。
この例では、width、height、そして depth プロパティがある三次元の長方形の箱を表す Cuboid と言う新しい構造体を定義しています。この構造体にはまた、volume と言う読み取り専用の計算されたプロパティもあり、立方体の現在の体積を計算して返します。width、height そして depth のどの値が、特定の volume の値に使用されるかに関してあいまいなので、volume が設定可能である事をこれは意味しません。それにも関わらず、Cuboid が現在の計算された体積を外部ユーザーが発見できるようにするために読み取り専用の計算されたプロパティを提供するのは、便利です。
プロパティ監視者は、プロパティの値の変化を監視し、対応します。プロパティ監視者は、新しい値がプロパティの現在の値と同じであっても、プロパティの値が設定されるたびに呼びだされます。
以下の場所にプロパティ監視者を追加できます。
継承したプロパティの場合、サブクラスでそのプロパティをオーバーライドすることにより、プロパティ監視者を追加します。定義した計算されたプロパティの場合、監視者を作成しようとする代わりに、プロパティのセッタを使用して値の変更を監視し、応答します。プロパティのオーバーライドについては、オーバーライド(上書き) で説明しています。
プロパティのこれらの監視者のいずれかまたは両方を定義するオプションがあります:
willSet 監視者を実装する場合、それには定数パラメータとして新しいプロパティ値が渡されます。willSet の実装の一部として、このパラメータの名前を指定できます。実装内でパラメータ名と括弧を書かない場合、パラメータは newValue のデフォルトのパラメータ名で利用できるようになります。
同様に、didSet 監視者を実装する場合、それには古いプロパティ値を含む定数パラメータが渡されます。パラメータに名前を付けるか、または oldValue のデフォルトのパラメータ名を使用できます。独自の didSet 監視者内のプロパティに値を割り当てる場合は、割り当てた新しい値は、ちょうど設定されたものを置き換えます。
ここでアクションの willSet と didSet の例を挙げます。以下の例では、人が歩く時のステップの総数を追跡する StepCounter と言う新しいクラスを定義しています。このクラスは、彼らの日常生活の間に、人の運動を追跡するために、歩数計またはその他の歩数カウンターからの入力データで使用されます。
StepCounter クラスは、Int 型の totalSteps プロパティを宣言します。これは willSet と didSet 監視者のある、格納されたプロパティです。
プロパティが新しい値を代入された時はいつでも totalSteps の willSet と didSet 監視者が呼び出されます。これは、新しい値が現在の値と同じ場合でも同様です。
この例の willSet 監視者は、今後の新しい値のために newTotalSteps と言うカスタムパラメータ名を使用します。この例では、設定しようとする値を単に印刷します。
totalSteps の値が更新された後 didSet 監視者が呼び出されます。それは、古い値に対して totalSteps の新しい値を比較します。総ステップ数が増加していた場合、どれだけ多く新たなステップが踏まれたかを示すメッセージが印刷されます。didSet 監視者は、古い値にカスタムパラメータ名を提供せず、代わりに oldValue のデフォルト名が使用されます。
プロパティラッパーは、プロパティの格納方法を管理するコードとプロパティを定義するコードの間に分離層を追加します。たとえば、スレッドセーフのチェックを提供するプロパティがある場合、または基になるデータをデータベースに格納する場合、すべてのプロパティにそのコードを記述しなければなりません。プロパティラッパーを使用する場合は、ラッパーを定義するときに管理コードを 1 回記述すれば、それを複数のプロパティに適用して管理コードをその時再利用できます。
プロパティラッパーを定義するには、wrappedValue プロパティを定義する構造体、列挙型、またはクラスを作成して下さい。以下のコードでは、TwelveOrLess 構造体により、ラップする値に常に 12 以下の数値が含まれることが保証されます。より大きい数値を格納するように要求すると、代わりに 12 が格納されます。
セッタは新しい値が 12 以下であることを保証し、ゲッタは格納された値を返します。
上記の例の number の宣言は、変数を private としてマークしており、これにより TwelveOrLess の実装でのみ number が使用される事を保証しています。他の場所で記述されたコードは、wrappedValue のゲッタとセッタを使用して値にアクセスし、number を直接使用することはできません。private の詳細については、アクセス制御 を参照してください。
属性としてプロパティの前にラッパーの名前を書くことで、プロパティにラッパーを適用します。 TwelveOrLess プロパティラッパーを使用して、その次元が常に 12 以下であることを保証する四角形を格納する構造体を以下に示します。
height プロパティと width プロパティは、TwelveOrLess.number をゼロに設定する TwelveOrLess の定義から初期値を取得します。TwelveOrLess のセッタは 10 を有効な値として扱うので、rectangle.height に数値 10 を格納すると、記述どおりに処理されます。ただし、24 は TwelveOrLess が許可するよりも大きいため、24 を格納しようとすると、最終的に、許可されている最大値である 12 に、rectangle.height は代わりに設定されます。
ラッパーをプロパティに適用すると、コンパイラはラッパーに保管場所を提供するコードと、ラッパーを介してプロパティへのアクセスを提供するコードを合成します。(プロパティラッパーはラップされた値を格納するのに責任があるため、その合成コードはありません。) 特別な属性構文を利用せずに、プロパティラッパーの動作を使用するコードを記述できます。たとえば、属性として @TwelveOrLess を記述する代わりに、 TwelveOrLess 構造体のプロパティを明示的にラップする以前のコードリストの SmallRectangle のバージョンは以下のとおりです。
_height プロパティと _width プロパティには、プロパティラッパーのインスタンス TwelveOrLess が格納されます。height と width のゲッタとセッタは、wrappedValue プロパティへのアクセスをラップします。
上記の例のコードは、TwelveOrLess の定義で初期値に number を指定することにより、ラップされたプロパティの初期値を設定します。このプロパティラッパーを使用するコードは、TwelveOrLess でラップされるプロパティに異なる初期値を指定できません。たとえば、SmallRectangle の定義では、height または width の初期値を指定できません。初期値の設定やその他のカスタマイズをサポートするには、プロパティラッパーはイニシャライザを追加する必要があります。以下に、SmallNumber と言う TwelveOrLess の拡張バージョンを示します。これは、ラップされた値と最大値を設定するイニシャライザを定義します。
SmallNumber の定義には、init( )、init(wrappedValue:)、および init(wrappedValue:maximum:) の 3 つのイニシャライザが含まれています。これらを使用して、以下の例ではラップされた値と最大値を設定します。初期化とイニシャライザの構文の詳細については、初期化 を参照してください。
ラッパーをプロパティに適用し、初期値を指定しない場合、Swift は init( ) イニシャライザを使用してラッパーをセットアップします。例えば:
SmallNumber( ) を呼び出すことにより、height と width をラップする SmallNumber のインスタンスが作成されます。イニシャライザ内のコードは、ゼロと 12 のデフォルト値を使用して、初期のラップされた値と初期最大値を設定します。SmallRectangle 内で TwelveOrLess を使用した以前の例のように、プロパティラッパーはすべての初期値をなお提供します。その例とは異なり、SmallNumbe はまた、プロパティの宣言の一部としてこれらの初期値の書き込みもサポートしています。
プロパティの初期値を指定すると、Swift は init(wrappedValue:) イニシャライザを使用してラッパーをセットアップします。例えば:
ラッパーを使用してプロパティ上に = 1 と記述すると、それは init(wrappedValue:) イニシャライザへの呼び出しに変換されます。height と width をラップする SmallNumber のインスタンスは、 SmallNumber(wrappedValue:1) を呼び出すことによって作成されます。イニシャライザは、ここで指定されたラップされた値を使用し、デフォルトの最大値の 12 を使用します。
カスタム属性の後に括弧で引数を記述すると、Swift はそれらの引数を受け入れるイニシャライザを使用してラッパーをセットアップします。たとえば、初期値と最大値を指定すると、Swift は init(wrappedValue:maximum:) イニシャライザを使用します。
height をラップする SmallNumber のインスタンスは SmallNumber(wrappedValue: 2、maximum:5) を呼び出すことによって作成され、width をラップするインスタンスは SmallNumber(wrappedValue: 3、maximun: 4) を呼び出すことによって作成されます。
プロパティラッパーに引数を含めることにより、ラッパーで初期状態を設定したり、ラッパーの作成時に他のオプションを渡すことができます。この構文は、プロパティラッパーを使用する最も一般的な方法です。属性に必要な全ての引数を提供すると、それらがイニシャライザに渡されます。
プロパティラッパー引数を含める場合、代入を使用して初期値を指定することもできます。Swift は代入を wrappedValue 引数のように扱い、含めた引数を受け入れるイニシャライザを使用します。例えば:
height をラップする SmallNumber のインスタンスは、SmallNumber(wrappedValue: 1) を呼び出して作成され、これは、デフォルトの最大値 12 を使用します。width をラップするインスタンスは、SmallNumber(wrappedValue: 2、maximum: 9) を呼び出して作成されます。
ラップされた値に加えて、プロパティラッパーは、投影値 を定義することで追加機能を公開できます。たとえば、データベースへのアクセスを管理するプロパティラッパーは、その投影値で flushDatabaseConnection( ) メソッドを公開できます。投影値の名前は、ドル記号 ($) で始まることを除いて、ラップされた値と同じです。コードでは $ で始まるプロパティを定義できないため、投影値は定義したプロパティに干渉することは決してありません。
上記の SmallNumber の例で、プロパティを大きすぎる数値に設定しようとすると、プロパティラッパーは数値を調整してから格納します。以下のコードは、projectedValue プロパティをSmallNumber 構造体に追加して、プロパティが新しい値を格納する前にプロパティラッパーの新しい値を調整したかどうかを追跡します。
someStructure.$someNumber と書くと、ラッパーの投影値にアクセスします。4 などの小さな数値を格納した後、 someStructure.$someNumber の値は false です。ただし、55 などの大きすぎる数値を格納しようとすると、投影値は true になります。
プロパティラッパーは、その投影値として任意の型の値を返すことができます。この例では、プロパティラッパーは 1 つの情報 (数値が調整されたかどうか) のみを公開するので、そのブール値を投影値として公開します。より多くの情報を公開する必要があるラッパーは、他の型のインスタンスを返すか、self を返して、ラッパーのインスタンスをその投影値として公開できます。
プロパティのゲッタやインスタンスメソッドなど、型の一部であるコードから投影値にアクセスする場合、他のプロパティにアクセスするのと同じように、プロパティ名の前の self. を省略できます。以下の例のコードは、height と width の周りのラッパーの投影値を $height と $width として参照しています。
プロパティラッパー構文は、ゲッタとセッタを持つプロパティの単なる構文上のシュガーなので、height と width にアクセスすることは、他のプロパティにアクセスするのと同じように動作します。たとえば、resize(to:) のコードは、それらのプロパティラッパーを使用して height と width にアクセスします。resize(to: .large) を呼び出すと、.large の switch の case は、長方形の高さと幅を 100 に設定します。ラッパーは、これらのプロパティの値が 12 を超えないようにし、投影値を true に設定し、値を調整したという事実を記録します。resize(to:) の最後で、return 文は $height と $width をチェックして、プロパティラッパーが height または width を調整したかどうかを判断します。
計算する、監視するプロパティの、上記で説明した機能はまた グローバル変数 と ローカル変数 にも利用可能です。グローバル変数は全ての関数、メソッド、クロージャ、または型のコンテキストの外で定義されている変数です。ローカル変数は、関数、メソッド、またはクロージャのコンテキスト内で定義されている変数です。
以前の章で遭遇したグローバルとローカル変数は、すべて 格納された変数 です。格納された変数は、格納されたプロパティと同様に、特定の型の値の記憶域を提供し、その値を設定し、取得することを可能にします。
ただし、計算された変数 を定義することもできますし、グローバルまたはローカルの範囲で、格納された変数の監視者も定義できます。計算された変数は、値を格納するよりも計算し、計算されたプロパティと同じように書かれます。
プロパティラッパーはローカルの格納された変数に適用できますが、グローバル変数または計算された変数には適用できません。たとえば、以下のコードでは、myNumber は SmallNumber をプロパティラッパーとして使用します。
プロパティに SmallNumber を適用する場合と同様に、myNumber の値を 10 に設定することは有効です。プロパティラッパーは 12 より高い値を許可しないため、myNumber を 24 ではなく 12 に設定します。
インスタンス・プロパティは、特定の型のインスタンスに属しているプロパティです。その型の新しいインスタンスを作成するたびに、それは他のインスタンスとは別のプロパティ値の独自のセットを持っています。
また、その型のいずれのインスタンスにでもない、型自体に属しているプロパティを定義することもできます。その型のインスタンスをどれだけ多くあなたが作成したかに関係なく、これらのプロパティのいずれか一つのコピーしかありません。これらの種類のプロパティは、型プロパティ と呼ばれます。
型プロパティは、特定の型の すべての インスタンスへの普遍的な値を定義し、すべてのインスタンスが (C の静的定数のように) 使える定数プロパティのように、またはその型の全てのインスタンスに対するグローバルな値を格納する (C の静的変数のように) 変数プロパティとして、有用です。
格納された型プロパティは、変数または定数にすることができます。計算された型プロパティは常に計算されたインスタンスのプロパティと同じように、変数プロパティとして宣言されます。
C 言語 と Objective-C では、グローバル 静的変数として型に関連した静的定数と変数を定義できます。Swift では、しかし、型プロパティは、型の外の中括弧の中に、型の定義の一部として書かれ、各々の型プロパティはサポートしている型に明示的にスコープされます。
static キーワードで型プロパティを定義して下さい。クラス型の計算された型プロパティでは、スーパークラスの実装をサブクラスがオーバーライド(上書き)するのを許す代わりに class キーワードを使用できます。以下の例で、格納された型プロパティと、計算された型プロパティの構文を示します。
型プロパティは、ちょうどインスタンス・プロパティのように、ドット構文で照会、設定できます。しかし、型プロパティはその型のインスタンスではなく、型 により照会、設定されます。例えば:
以下の例は、オーディオチャンネル数のオーディオレベルメーターをモデルにした構造体の一部として2つの格納された型プロパティを使用しています。各チャンネルには、0 から 10 までの整数のオーディオレベルがあります。
下の図は、これら2つのオーディオチャンネルが、ステレオのオーディオ・レベルメーターをモデル化して組み合わせられる所を示しています。チャンネルのオーディオレベルが 0 の場合、そのチャンネルの光はいずれも点灯しません。オーディオレベルが 10 の場合、そのチャンネルの光は全て点灯します。この図では、左チャンネルは現在 9 のレベルであり、右チャンネルは現在 7 のレベルです。
上述のオーディオチャンネルを AudioChannel 構造体のインスタンスによって表わします。
AudioChannel 構造体は、その機能をサポートするために、2つの格納された型プロパティを定義しています。まず、thresholdLevel は、オーディオレベルが取りうる最大の閾値を定義しています。これは、すべての AudioChannel インスタンスに対する 10 の定数値です。オーディオ信号が 10 よりも高い値になる場合には、この閾値にフタをされます (後述)。
第二の型プロパティは、maxInputLevelForAllChannels という変数に格納されたプロパティです。これは、全て の AudioChannel インスタンスによって受信された最大入力値を追跡します。これは、初期値 0 で始まります。
AudioChannel 構造体はまた、0 から 10 のスケールでチャンネルの現在のオーディオレベルを表す currentLevel という、格納されたインスタンスプロパティも定義しています。
currentLevel プロパティには、それが設定されるたびに currentLevel の値を確認する didSet プロパティの監視者があります。この監視者は2つのチェックを実行します。
ステレオサウンドシステムのオーディオレベルを表すために、leftChannel と rightChannel という二つの新しいオーディオチャンネルを作成するために AudioChannel 構造体を使用します。
左 チャンネルの currentLevel を 7 に設定すると、maxInputLevelForAllChannels の型プロパティが 7 に等しく更新されていることがわかるでしょう。
右 チャンネルの currentLevel に 11 を設定しようとすると、右チャンネルの currentLevel プロパティが 10 の最大値でフタをされ、maxInputLevelForAllChannels 型プロパティは、10 に等しく更新されます。