明確な割り当てアサーション(definite assignment assertion)
明確な割り当てアサーションは、変数やプロパティが確実に初期化されていることをTypeScriptのコンパイラに伝える演算子です。
strictNullChecksと変数の初期化エラー
TypeScriptはコンパイラオプションstrictNullChecksがtrueのとき、初期化されていない変数を参照した際にエラーを出します。
tsnum : number;Variable 'num' is used before being assigned.2454Variable 'num' is used before being assigned.console .log (* 2); num 
tsnum : number;Variable 'num' is used before being assigned.2454Variable 'num' is used before being assigned.console .log (* 2); num 
変数の初期化が明らかに関数内で行われている場合でも、コンパイラは変数が初期化されていないとエラーを出します。
tsnum : number;initNum (); // 関数内でnumを初期化しているが…Variable 'num' is used before being assigned.2454Variable 'num' is used before being assigned.console .log (* 2); num functioninitNum () {num = 2;}
tsnum : number;initNum (); // 関数内でnumを初期化しているが…Variable 'num' is used before being assigned.2454Variable 'num' is used before being assigned.console .log (* 2); num functioninitNum () {num = 2;}
strictPropertyInitializationとプロパティの初期化エラー
TypeScriptでは次のコンパイラオプションの両方がtrueのとき、クラスのプロパティが初期化されていないとエラーを出します。
tsFoo {Property 'num' has no initializer and is not definitely assigned in the constructor.2564Property 'num' has no initializer and is not definitely assigned in the constructor.: number; num }
tsFoo {Property 'num' has no initializer and is not definitely assigned in the constructor.2564Property 'num' has no initializer and is not definitely assigned in the constructor.: number; num }
TypeScriptコンパイラは、プロパティ定義またはconstructorでプロパティが初期化されるかを見ています。しかし、constructor以外のメソッドで初期化されるところまでは追いかけません。たとえば、次例のnum3は実際は初期化されるものの、コンパイラは初期化がされていないと警告を出します。
tsFoo {num1 : number = 1; // 初期化しているnum2 : number;Property 'num3' has no initializer and is not definitely assigned in the constructor.2564Property 'num3' has no initializer and is not definitely assigned in the constructor.: number; num3 constructor() {this.num2 = 1; // 初期化しているthis.initNum3 (); // num3を初期化している}initNum3 () {this.num3 = 1;}}
tsFoo {num1 : number = 1; // 初期化しているnum2 : number;Property 'num3' has no initializer and is not definitely assigned in the constructor.2564Property 'num3' has no initializer and is not definitely assigned in the constructor.: number; num3 constructor() {this.num2 = 1; // 初期化しているthis.initNum3 (); // num3を初期化している}initNum3 () {this.num3 = 1;}}
明確な割り当てアサーションを使う
変数やプロパティの初期化が確実に行われていることをコンパイラに伝えるには、明確な割り当てアサーションを使います。変数宣言の変数名やプロパティ名のあとに!を書きます。
tsnum !: number;// ^明確な割り当てアサーションinitNum ();console .log (num * 2); // エラーにならないfunctioninitNum () {num = 2;}
tsnum !: number;// ^明確な割り当てアサーションinitNum ();console .log (num * 2); // エラーにならないfunctioninitNum () {num = 2;}
tsFoo {num !: number;// ^明確な割り当てアサーション}
tsFoo {num !: number;// ^明確な割り当てアサーション}
非Nullアサーション
別の方法として、非Nullアサーション(non-null assertion)を使う方法もあります。この場合は、変数を参照するコードにて、変数のあとに!を書きます。
tsnum : number;initNum ();console .log (num ! * 2); // エラーにならない// ^非NullアサーションfunctioninitNum () {num = 2;}
tsnum : number;initNum ();console .log (num ! * 2); // エラーにならない// ^非NullアサーションfunctioninitNum () {num = 2;}
より安全なコードを書くには
明確な割り当てアサーションと非Nullアサーションは、型の安全性を保証する責任をコンパイラからプログラマに移すものです。そして、型に関してはコンパイラより人間のほうがミスをしやすいです。なので、こうしたアサーションはできる限り使わないほうが安全性は高いです。
たとえば、上の例であればinitNumの戻り値をnumに代入するほうが、より安全なコードになります。
tsnum : number;num =initNum ();console .log (num * 2);functioninitNum () {return 2;}
tsnum : number;num =initNum ();console .log (num * 2);functioninitNum () {return 2;}
他にも、numがnumber型であるかを型ガードでチェックする方法もあります。
tsnum : number | undefined;initNum ();// 型ガードif (typeofnum === "number") {console .log (num * 2);}functioninitNum () {num = 2;}
tsnum : number | undefined;initNum ();// 型ガードif (typeofnum === "number") {console .log (num * 2);}functioninitNum () {num = 2;}
このようにアサーションに頼らない方法はないかを先に検討することをお勧めします。その上で、どうしてもというときにアサーションを使うようにしましょう。フレームワークやライブラリの都合で、やむを得ない場合もあります。
学びをシェアする
・明確な割り当てアサーションは、変数初期化が確実であるとTypeScriptのコンパイラに伝える
・変数名のあとに!を書く
・型安全の責任をコンパイラからプログラマに移すものなので、使わない方法を先に検討する
・どうしようもないときに使う
『サバイバルTypeScript』より
関連情報
📄️ strictNullChecks
null・undefinedのチェックを厳しくする
📄️ strictPropertyInitialization
クラスプロパティの初期化を必須にする