読み込み中...

TypeScriptでinstanceof演算子をマスターするたったの10の方法

TypeScriptのinstanceof演算子のイラスト付き解説 TypeScript
この記事は約23分で読めます。

【サイト内のコードはご自由に個人利用・商用利用いただけます】

この記事では、プログラムの基礎知識を前提に話を進めています。

説明のためのコードや、サンプルコードもありますので、もちろん初心者でも理解できるように表現してあります。

本記事のサンプルコードを活用して機能追加、目的を達成できるように作ってありますので、是非ご活用ください。

※この記事は、一般的にプロフェッショナルの指標とされる『実務経験10,000時間以上』を満たす現役のプログラマチームによって監修されています。

※Japanシーモアは、常に解説内容のわかりやすさや記事の品質に注力しております。不具合、分かりにくい説明や不適切な表現、動かないコードなど気になることがございましたら、記事の品質向上の為にお問い合わせフォームにてご共有いただけますと幸いです。
(送信された情報は、プライバシーポリシーのもと、厳正に取扱い、処分させていただきます。)

はじめに

TypeScriptは、JavaScriptのスーパーセットとして、型システムを導入することで、コードの品質とメンテナンス性を向上させることができる言語です。

TypeScriptの中で、特定のオブジェクトが特定のクラスやコンストラクタのインスタンスであるかを確認するための非常に強力なツールが「instanceof演算子」です。

この記事では、このinstanceof演算子の基本的な使用方法から、さまざまな応用例まで、詳細に解説していきます。

●TypeScriptのinstanceof演算子とは

TypeScriptにおけるinstanceofは、特定のオブジェクトが特定のクラスやコンストラクタのインスタンスであるかをチェックするための二項演算子です。

JavaScriptでも使用されるこの演算子は、TypeScriptでは型チェックの際にさらなる力を発揮します。

これにより、動的なコードの中でオブジェクトの型を安全にチェックすることができます。

例えば、JavaScriptではオブジェクトが配列であるかどうかを確認するのにArray.isArray()を使用する場面も多いですが、TypeScriptではinstanceofを使用して同様のチェックを行うことができます。

しかし、これは基本的な使用例に過ぎません。

TypeScriptの強力な型システムをフルに活用するためには、instanceofの使い方を深く理解することが重要です。

○instanceofの基本的な概念

instanceof演算子は、左のオペランドに与えられたオブジェクトが、右のオペランドに与えられたコンストラクタのインスタンスであるかどうかを確認します。

戻り値は真偽値(trueまたはfalse)となります。

例を見てみましょう。

class Animal {}
class Dog extends Animal {}

let dog = new Dog();

console.log(dog instanceof Dog); // true
console.log(dog instanceof Animal); // true

このコードでは、DogクラスがAnimalクラスを継承しています。

したがって、dogオブジェクトはDogクラスのインスタンスであり、Animalクラスのインスタンスでもあります。

これを確認するために、instanceof演算子を使用しています。

また、組み込みのオブジェクトやクラスに対してもinstanceofを使用することができます。

let numbers = [1, 2, 3];

console.log(numbers instanceof Array); // true
console.log(numbers instanceof Object); // true

このコードでは、numbersという名前の配列がArrayクラスのインスタンスであり、Objectクラスのインスタンスでもあることを表しています。

●instanceof演算子の正確な使い方

TypeScriptにおける「instanceof」とは、特定のオブジェクトがあるクラスやコンストラクタのインスタンスであるかを判定するための演算子です。

JavaScriptでも存在するこの演算子は、TypeScriptにおいても非常に役立つ型判定のツールとなっています。

特に、カスタムクラスや組み込みオブジェクト、さらにはインターフェースと組み合わせて利用することで、強力な型チェックを実現することができます。

ここでは、TypeScriptにおける「instanceof」演算子の正確な使い方を、実例を交えて解説します。

○サンプルコード1:基本的な型チェック

下記のサンプルコードは、基本的な型チェックを行うものです。

// 簡単なクラスの定義
class Animal {
  name: string;
  constructor(name: string) {
    this.name = name;
  }
}

class Dog extends Animal {
  bark() {
    return "わんわん";
  }
}

const pet = new Dog("ポチ");

// instanceof演算子を用いた型チェック
if (pet instanceof Dog) {
  console.log(pet.bark()); // わんわん
}

このコードでは、Animalクラスを定義しており、そのサブクラスとしてDogクラスも定義されています。

Dogクラスのインスタンスpetを作成した後、instanceof演算子を使用して、petDogクラスのインスタンスであるかどうかを判定しています。

この判定が真であれば、Dogクラス特有のメソッドであるbarkを呼び出しています。

このコードを実行すると、Dogクラスのインスタンスであるため、コンソールには”わんわん”と出力されます。

○サンプルコード2:カスタムクラスでの利用方法

TypeScriptでは、instanceof演算子はカスタムクラスとともに使用することも可能です。

この演算子を使うことで、特定のオブジェクトが特定のクラスのインスタンスであるかをチェックすることができます。

instanceof演算子をカスタムクラスで利用する際のサンプルコードを紹介します。

class Animal {
    name: string;
    constructor(name: string) {
        this.name = name;
    }
}

class Dog extends Animal {
    bark() {
        console.log("ワンワン");
    }
}

const pet = new Dog("シロ");

if (pet instanceof Dog) {
    pet.bark();  // ワンワン と出力される
}

このコードでは、まずAnimalという基底クラスを定義しています。そして、DogというクラスがAnimalを継承しています。

Dogクラスには、barkという犬が吠えるメソッドが追加されています。

petという変数に、Dogクラスのインスタンスを代入しています。

その後、instanceof演算子を使用してpetDogクラスのインスタンスであるかをチェックしています。

もし、Dogのインスタンスであれば、barkメソッドを呼び出し、”ワンワン”という音を出力します。

このコードを実行すると、”ワンワン”という出力が得られます。

なぜなら、petDogクラスのインスタンスであるためです。

○サンプルコード3:組み込みオブジェクトとの組み合わせ

TypeScriptの強力な型システムをフルに活用するためには、様々な型判定手法を理解する必要があります。

instanceof演算子はその中でも非常に実用的なものの一つです。

特に、組み込みオブジェクトやカスタムクラスと組み合わせることで、その真価を発揮します。

ここでは、組み込みオブジェクトとinstanceof演算子を組み合わせた型判定の方法について、実例を交えて詳しく解説していきます。

TypeScriptでは、JavaScriptの組み込みオブジェクトも、もちろん型チェックの対象となります。

例えば、Array、Date、RegExpなど、これらの組み込みオブジェクトを効果的に型チェックする方法として、instanceof演算子が非常に役立ちます。

組み込みオブジェクトとしてのArrayとDateをinstanceof演算子を使用して型チェックするサンプルコードを紹介します。

class CustomDate {
    constructor(public date: Date) {}
}

function checkType(value: any): string {
    // このコードではArrayのインスタンスかどうかを判定しています
    if (value instanceof Array) {
        return '配列です';
    }
    // このコードではDateのインスタンスかどうかを判定しています
    else if (value instanceof Date) {
        return '日付オブジェクトです';
    }
    // このコードではCustomDateのインスタンスかどうかを判定しています
    else if (value instanceof CustomDate) {
        return 'カスタム日付オブジェクトです';
    } else {
        return 'その他の型です';
    }
}

const sampleArray = [1, 2, 3];
const sampleDate = new Date();
const customDateObj = new CustomDate(new Date());

console.log(checkType(sampleArray));  // 配列です
console.log(checkType(sampleDate));   // 日付オブジェクトです
console.log(checkType(customDateObj)); // カスタム日付オブジェクトです

このコードを実行すると、それぞれの組み込みオブジェクトやカスタムクラスのインスタンスが正しく判定され、期待されるメッセージがコンソールに出力されます。

具体的には、「配列です」、「日付オブジェクトです」、「カスタム日付オブジェクトです」という文字列が順に表示されます。

○サンプルコード4:インターフェースを活用した型判定

TypeScriptでは、より柔軟な型システムを持っており、その中でインターフェースは非常に重要な役割を果たしています。

インターフェースを利用することで、特定の形状や契約を持つオブジェクトを表現することができます。

そして、instanceof演算子を使って、そのインターフェースを実装したオブジェクトの型チェックを行うことができます。

しかし、ここで注意が必要なのは、instanceofはクラスのインスタンスに対してのみ使用可能であり、インターフェースに直接使用することはできません。

このため、インターフェースを活用した型判定を行う際は、そのインターフェースを実装するクラスを定義し、そのクラスのインスタンスに対してinstanceofを使用します。

次に、この方法を表すサンプルコードを見てみましょう。

// インターフェースの定義
interface Animal {
    name: string;
    makeSound(): void;
}

// Animalインターフェースを実装するDogクラスを定義
class Dog implements Animal {
    name: string;
    constructor(name: string) {
        this.name = name;
    }
    makeSound() {
        console.log('ワンワン');
    }
}

// Animalインターフェースを実装するCatクラスを定義
class Cat implements Animal {
    name: string;
    constructor(name: string) {
        this.name = name;
    }
    makeSound() {
        console.log('ニャー');
    }
}

const dog = new Dog('シロ');
const cat = new Cat('ミケ');

if (dog instanceof Dog) {
    dog.makeSound();  // ワンワン
}

if (cat instanceof Cat) {
    cat.makeSound();  // ニャー
}

このコードでは、まずAnimalというインターフェースを定義しています。

このインターフェースはnameプロパティとmakeSoundメソッドを持っています。

次に、このインターフェースを実装するDogクラスとCatクラスを定義しています。

そして、これらのクラスのインスタンスを作成し、それぞれのインスタンスが指定したクラスのものであるかをinstanceofを使ってチェックしています。

このコードを実行すると、それぞれの動物の鳴き声が出力されます。

●instanceofの応用例

TypeScriptのプログラミングの中で、特に複雑な型チェックやオブジェクト指向の機能を利用する場面でinstanceof演算子が非常に有効です。

実際のコードを見ながら、instanceofの応用的な使用方法を探ります。

○サンプルコード5:動的な関数の振る舞い制御

TypeScriptで動的に関数の振る舞いを制御する場面は多いです。

例えば、異なるクラスのインスタンスを受け取る関数があり、それぞれのクラスに応じて異なる処理をしたい場合、instanceofを活用することで簡潔に条件分岐を行うことができます。

下記のサンプルコードでは、Animalクラスを基底として、DogとCatの2つのクラスを継承します。

そして、makeSound関数では、渡されるインスタンスに応じて、それぞれの動物の鳴き声を出力するように制御します。

class Animal {
    // 基底クラス
}

class Dog extends Animal {
    bark() {
        console.log("ワンワン");
    }
}

class Cat extends Animal {
    meow() {
        console.log("ニャー");
    }
}

function makeSound(animal: Animal) {
    if (animal instanceof Dog) {
        animal.bark(); // このコードではDogクラスのインスタンスを判定しています。
    } else if (animal instanceof Cat) {
        animal.meow(); // このコードではCatクラスのインスタンスを判定しています。
    } else {
        console.log("不明な動物の音");
    }
}

const dog = new Dog();
const cat = new Cat();

makeSound(dog); // このコードを実行すると、Dogクラスのbarkメソッドが呼ばれ、"ワンワン"と出力されます。
makeSound(cat); // このコードを実行すると、Catクラスのmeowメソッドが呼ばれ、"ニャー"と出力されます。

上記の例で重要なのは、instanceof演算子を使用してDogクラスとCatクラスのインスタンスを判定し、それぞれの動物に応じた鳴き声を出力する部分です。

このようにinstanceofを使用することで、型安全にオブジェクトの実際のクラスを判定し、そのクラスに特有のメソッドやプロパティを使用することができます。

○サンプルコード6:条件分岐による特定のクラスメソッドの呼び出し

TypeScriptを使用する上で、instanceof演算子はとても便利な型チェックのツールです。

特に、条件分岐を活用して特定のクラスのメソッドを呼び出す際に、この演算子は非常に有効です。

その方法について詳しく見ていきましょう。

まず、次のコードをご覧ください。

class Animal {
    sound(): string {
        return "Some sound";
    }
}

class Dog extends Animal {
    sound(): string {
        return "Woof!";
    }
}

class Cat extends Animal {
    sound(): string {
        return "Meow!";
    }
}

function makeSound(animal: Animal) {
    if (animal instanceof Dog) {
        console.log(animal.sound());
    } else if (animal instanceof Cat) {
        console.log(animal.sound());
    } else {
        console.log(animal.sound());
    }
}

const myDog = new Dog();
const myCat = new Cat();
makeSound(myDog);  // Woof!
makeSound(myCat);  // Meow!

このコードでは、Animalという親クラスがあり、DogとCatという2つのサブクラスが存在します。

それぞれのクラスは、soundというメソッドを持ち、動物の鳴き声を返すようになっています。

makeSound関数は、Animal型の引数を受け取り、その引数が実際にはどのクラスのインスタンスなのかを、instanceof演算子を使って判定しています。

Dogのインスタンスであれば”Woof!”、Catのインスタンスであれば”Meow!”と、それぞれの動物の鳴き声をコンソールに表示します。

上記のコードを実行すると、makeSound(myDog);の部分では”Woof!”が、makeSound(myCat);の部分では”Meow!”がコンソールに表示される結果になります。

○サンプルコード7:エラーハンドリングとの連携

TypeScriptを使用するとき、instanceof演算子とエラーハンドリングは密接な関係があります。

特に、異なる型のオブジェクトを扱っている場合、不適切な操作を行うとエラーが発生する可能性があります。

そのため、instanceof演算子を使って適切に型チェックを行い、適切なエラーハンドリングを組み合わせることで、安全なコードを記述することができます。

具体的な方法を、次のサンプルコードで確認してみましょう。

// サンプルのクラス定義
class Animal {
    name: string;
    constructor(name: string) {
        this.name = name;
    }

    speak() {
        console.log(`${this.name} makes a noise.`);
    }
}

class Dog extends Animal {
    bark() {
        console.log(`${this.name} barks.`);
    }
}

// エラーハンドリングを組み込んだ関数
function handleAnimalSpeak(animal: Animal) {
    if (animal instanceof Dog) {
        // Dogクラスのメソッドを安全に呼び出す
        animal.bark();
    } else {
        animal.speak();
    }
}

try {
    const myDog = new Dog('Rover');
    handleAnimalSpeak(myDog); // Rover barks.

    const myAnimal = new Animal('Lion');
    handleAnimalSpeak(myAnimal); // Lion makes a noise.
} catch (error) {
    console.error("エラーが発生しました:", error);
}

このコードでは、まずAnimalクラスとそのサブクラスであるDogクラスを定義しています。

そして、handleAnimalSpeak関数内でinstanceof演算子を使用して、引数で受け取ったオブジェクトがDogのインスタンスかどうかを確認します。

この確認によって、Dogのインスタンスであればbarkメソッドを、そうでなければAnimalspeakメソッドを呼び出します。

実際に、この関数をtryブロック内で実行してみると、myDogDogのインスタンスであるためRover barks.という結果が出力され、myAnimalAnimalのインスタンスであるためLion makes a noise.という結果が出力されます。

●注意点と対処法

TypeScriptでinstanceof演算子を使用する際、さまざまな利点がありますが、一方で知っておくべき注意点も存在します。

ここでは、そのような注意点とその対処法を取り上げます。

○instanceofの限界と注意点

TypeScriptのinstanceof演算子は強力な型チェックツールの1つとして知られています。

しかしその強力さと引き換えに、いくつかの限界や注意すべきポイントが存在します。

❶プリミティブ値のチェックには適していません

TypeScriptのinstanceof演算子は、オブジェクトのインスタンスをチェックするためのものです。

これは、プリミティブ型(数値、文字列、ブーリアンなど)の値のチェックには不適切であることを意味します。

❷異なるコンテキスト間でオブジェクトを共有する場合、期待通りに動作しないことがあります

例えば、iframeやWeb Workerなど、異なる実行コンテキストを持つ場合、同じコンストラクタ関数でも異なるものとして認識されることがあります。

❸カスタムクラスの拡張やプロトタイプチェーンが深い場合、挙動が複雑になることがあります

クラスの継承関係が複雑になると、期待した型と異なる結果を返すことがあります。

○サンプルコード8:instanceofの罠とその回避法

例として、次のようなクラスがあるとします。

class Animal {
  name: string;
  constructor(name: string) {
    this.name = name;
  }
}

class Cat extends Animal {
  meow() {
    return "にゃーん";
  }
}

class Dog extends Animal {
  bark() {
    return "ワンワン";
  }
}

下記のコードでは、Animalクラスのインスタンスをチェックしています。

const kitty = new Cat("キティ");

if (kitty instanceof Animal) {
  console.log(kitty.name); // これは期待通りに動作する
} else {
  console.log("これはAnimalのインスタンスではありません");
}

このコードでは、kittyAnimalクラスのインスタンスであるかをinstanceof演算子でチェックしています。

このコードを実行すると、”キティ”という名前が正しく出力されます。

しかし、CatDogをチェックする場面で問題が発生する可能性があります。

const kitty = new Cat("キティ");

if (kitty instanceof Dog) {
  console.log(kitty.bark());
} else {
  console.log("これはDogのインスタンスではありません");
}

このコードでは、kittyDogクラスのインスタンスであるかをinstanceof演算子でチェックしています。

このコードを実行すると、”これはDogのインスタンスではありません”という結果が得られます。

●カスタマイズ方法

TypeScriptにおけるinstanceof演算子は、オブジェクトが特定のクラスやコンストラクタのインスタンスであるかをチェックする際に非常に役立ちます。

しかし、それだけでなく、この演算子をカスタマイズし、より独自の型チェックを行う方法もあります。

これには、カスタム型ガードを作成する方法があります。このセクションでは、カスタム型ガードの作成方法を中心に解説していきます。

○サンプルコード9:カスタム型ガードの作成

TypeScriptでは、ユーザーが独自に型ガード関数を定義できます。

型ガード関数は、特定の型であるかをチェックする関数であり、特定の型であることを確認できる場合に、その型に型を絞り込むことができます。

// サンプルのクラスを定義
class Animal {
  name: string;
  constructor(name: string) {
    this.name = name;
  }
}

class Dog extends Animal {
  bark(): void {
    console.log(`${this.name}はワンワンと鳴きます`);
  }
}

// カスタム型ガードを定義
function isDog(obj: any): obj is Dog {
  return obj instanceof Dog;
}

const pet = new Dog("ポチ");

// カスタム型ガードを使用して型を絞り込む
if (isDog(pet)) {
  pet.bark();  // Dogのメソッドであるbark()が呼び出せる
}

このコードでは、まずAnimalクラスとそのサブクラスであるDogクラスを定義しています。

次に、isDogというカスタム型ガード関数を定義しています。

この関数は、引数として与えられたオブジェクトがDogクラスのインスタンスであるかをinstanceof演算子を使ってチェックし、その結果を返しています。

この型ガード関数を使えば、特定の型であることが確認できる場合にその型に絞り込みを行うことができます。

この例では、petがDogクラスのインスタンスであるかをisDog関数でチェックしています。

結果として、petがDogであると確認できた場合、Dogクラスのメソッドであるbark()を安全に呼び出すことができます。

カスタム型ガードを活用することで、instanceof演算子だけでなく、独自の条件での型チェックや型の絞り込みを行うことが可能になります。

これにより、プログラムの柔軟性や安全性を高めることができます。

このサンプルコードを実行すると、”ポチはワンワンと鳴きます”という出力が得られます。

この出力結果から、カスタム型ガード関数が正しく動作し、Dogクラスのインスタンスとしてpetを認識していることがわかります。

○サンプルコード10:ユーザー定義の型アサーションと連携

TypeScriptでは、型をチェックするための多くの方法が提供されています。

instanceof演算子はその中でも非常に有用ですが、実際のプロジェクトでは、より詳細な型のチェックや調整が必要になることも少なくありません。

ここでは、ユーザー定義の型アサーションとinstanceofを組み合わせる方法について、詳細に解説します。

まず、基本的なコードの例から見ていきましょう。

// サンプルのクラス定義
class Dog {
    bark() {
        return 'ワンワン!';
    }
}

class Cat {
    meow() {
        return 'ニャー!';
    }
}

// ユーザー定義の型アサーション関数
function isDog(obj: any): obj is Dog {
    return obj instanceof Dog;
}

const animal = new Dog();

if (isDog(animal)) {
    console.log(animal.bark());  // ワンワン!
} else {
    console.log('この動物は犬ではありません。');
}

このコードでは、DogCatという2つのクラスを定義しています。

そして、isDogというユーザー定義の型アサーション関数を作成しています。

この関数は、引数として与えられたオブジェクトがDogクラスのインスタンスであるかどうかを判定します。

このように、ユーザー定義の型アサーションを使用すると、特定のクラスやインターフェースに属するオブジェクトのみを対象とした処理を簡潔に記述することができます。

次に、上記のサンプルコードを実行するとどのような結果が得られるかを確認します。

上記のコードを実行すると、コンソールには「ワンワン!」と表示されます。

なぜなら、animalという変数にはDogクラスのインスタンスが格納されており、isDog関数によってその判定が行われ、真と評価されるためです。

まとめ

TypeScriptでのinstanceof演算子の使用は、開発者にとって非常に重要な技術の一つです。

この演算子を使うことで、オブジェクトの型やクラスを確認することができ、さまざまなシチュエーションで役立ちます。

この記事を通じて、TypeScriptでのinstanceof演算子の効果的な使用方法を理解できたでしょうか?

日常の開発での型チェックや型判定の際に、この演算子を活用し、より質の高いコードを書く手助けにしてください。