🎣

オブジェクト操作を自由自在に操るProxy

に公開

JavaScriptのProxyは、オブジェクトの基本的な操作(プロパティの参照、代入、関数の呼び出しなど)をインターセプト(途中で捕捉し横取り)し、その挙動をカスタマイズするための非常に強力な機能です。

オブジェクトへのアクセスを監視・制御したり、新しい動作を追加したりする際に大変役立ちます。

基本的な仕組み

Proxyは、ターゲットとなるオブジェクトと、そのオブジェクトに対する操作を定義するハンドラーオブジェクトの2つで構成されます。

const proxy = new Proxy(target, handler);
パラメータ 説明
target プロキシの対象となる元のオブジェクト
オブジェクト、配列、関数など、どんなものでも指定できます。
handler targetに対する操作を定義するオブジェクト。
この操作はトラップと呼ばれ、例えばプロパティの読み取り(get)や書き込み(set)、メソッドの呼び出し(apply)など、様々な操作に対するカスタムロジックを記述できます。

主なトラップの種類

handlerオブジェクトには、以下のような様々なトラップ(メソッド)を定義できます。

トラップ名 処理内容
get プロパティの読み取り
set プロパティの書き込み
apply 関数の呼び出し
construct new演算子によるインスタンス化

他にも多くのオブジェクト内部メソッドに連動するトラップがあります。詳細については、MDNのProxyドキュメント#オブジェクト内部メソッドをご参照ください。

活用例

Proxyは、様々な場面でその真価を発揮します。

  • データバインディングとリアクティビティ: Vue.js 3などで使用されているように、オブジェクトの変更を検知してUIを自動的に更新するシステムを構築できます。
  • バリデーション: オブジェクトのプロパティへの代入時に、値の型や範囲をチェックし、不正な値を防ぎます。
  • ロギングとデバッグ: オブジェクトへのアクセスや変更をログに出力し、デバッグを容易にします。
  • アクセスの制御: 特定のプロパティへのアクセスを制限したり、権限に応じて挙動を変えたりできます。
  • 仮想オブジェクト: 実際に存在しないプロパティへのアクセスを処理し、動的に値を生成するような仮想的なオブジェクトを作成できます。
  • パフォーマンスモニタリング: メソッドの実行時間などを計測し、パフォーマンスのボトルネックを特定します。

注意点

Proxyは非常に強力ですが、いくつか注意すべき点があります。

  • パフォーマンスオーバーヘッド: すべての操作を監視し別の処理を行うため、直接オブジェクトにアクセスするよりも若干のパフォーマンスコストが発生する可能性があります。
  • 複雑性: 多くのトラップを扱う場合、コードが複雑になることがあります。
  • 非透過性: プロキシされたオブジェクトは、元のオブジェクトとは異なる振る舞いをすることがあるため、デバッグ時に予期せぬ挙動に遭遇する可能性があります。

主なオブジェクトに対する操作(トラップ)の種類

ここからは、特によく使われるトラップについて、その詳細と具体的な利用例をご紹介します。

get(target, property, receiver): プロパティの読み取り

オブジェクトのプロパティにアクセスしようとしたときに呼び出されます。

パラメータ 説明
target プロキシの対象とした元のオブジェクト
property アクセスしようとしているプロパティ名(文字列またはSymbol)
receiver プロキシまたはプロキシを継承するオブジェクト(通常はプロキシ自身)

利用例:存在しないプロパティへのアクセスを処理する

const user = {
    name: '田中',
    age: 30
};

const userProxy = new Proxy(user, {
    get(target, property) {
        if (property in target) {
            return target[property];
        } else {
            console.warn(`プロパティ '${String(property)}' は存在しません。`);
            return undefined; // または適切なデフォルト値
        }
    }
});

console.log(userProxy.name);    // 田中
console.log(userProxy.age);     // 30
console.log(userProxy.address); // プロパティ 'address' は存在しません。 undefined

set(target, property, value, receiver): プロパティの書き込み

オブジェクトのプロパティに値を代入しようとしたときに呼び出されます。

パラメータ 説明
target プロキシの対象としたオブジェクト
property 代入しようとしているプロパティ名(文字列またはSymbol)
value 代入しようとしている値
receiver プロキシまたはプロキシを継承するオブジェクト

利用例:プロパティのバリデーション

const userProfile = {
    name: '田中',
    age: 30
};

const profileProxy = new Proxy(userProfile, {
    set(target, property, value) {
        if (property === 'age') {
            if (typeof value !== 'number' || value < 0) {
                console.error('年齢は0以上の数値でなければなりません。');
                return false; // 代入をキャンセル
            }
        }
        target[property] = value;
        return true; // 代入を許可
    }
});

profileProxy.age = 30;  // 正常に代入
console.log(profileProxy.age); // 30

profileProxy.age = -5;  // エラー出力: 年齢は0以上の数値でなければなりません。
console.log(profileProxy.age); // 30 (値は変更されない)

apply(target, thisArg, argumentsList): 関数の呼び出し

パラメータ 説明
target プロキシの対象とした関数
thisArg 呼び出し時のthisの値
argumentsList 呼び出し時に渡された引数の配列

利用例:関数呼び出しのログ出力

function sum(a, b) {
    return a + b;
}

const sumProxy = new Proxy(sum, {
    apply(target, thisArg, argumentsList) {
        console.log(`sum関数が呼び出されました。引数: ${argumentsList}`);
        return Reflect.apply(target, thisArg, argumentsList); // 元の関数を実行
    }
});

console.log(sumProxy(10, 20)); // sum関数が呼び出されました。引数: 10,20 \n 30

Reflectオブジェクトについて

Reflectオブジェクトは、Proxyハンドラー内で元の操作を安全かつ適切に実行するための静的メソッドを提供します。
apply()の例でReflect.apply が使用されています。

Reflectを使用することには、主に以下のメリットがあります。

  • thisの扱いが容易に: 多くのオブジェクト操作はthisコンテキストに依存しますが、Reflectを使うことで正しいthisが自動的に適用されます。
  • エラー処理: 操作が失敗した場合に適切な例外をスローします。
  • 簡潔なコード: 元の操作を直接呼び出すよりも、意図が明確になります。

Proxyを使用する際には、対応するReflectメソッドをセットで利用することが、より堅牢なコードを書く上で推奨されます。

まとめ

JavaScriptのProxyは、オブジェクトの動作を柔軟にカスタマイズし、高度な機能を実現するための強力なツールです。データバインディングやバリデーション、ロギングなど、様々な場面で活用できます。

少し慣れが必要な部分もありますが、使いこなせるようになると、JavaScriptでの開発の幅が大きく広がるはずです。ぜひ、ご自身のプロジェクトでProxyの力を試してみてくださいね!

リンク

https://developer.mozilla.org/ja/docs/Web/JavaScript/Reference/Global_Objects/Proxy

Discussion