Java の匿名内部クラスについて話す
多くの場合、クラス内で静的な Map または List を初期化し、クラスの内部メソッドで使用できるように定数値を保存する必要があります。
私たちの通常のアプローチは次のとおりです:
まず Map の静的変数を初期化します。
次に、静的ブロックに定数値を追加します。
Java コード
private final static Map<String, String> CONSTANT = new HashMap<String, String>(); static { CONSTANT.put("1", "one"); CONSTANT.put("2", "two"); }
実際、次のように記述することもできます:
Java コード
private final static Map<String, String> CONSTANT = new HashMap<String, String>() { { put("1", "one"); put("2", "two"); } };
このメソッドに慣れていない場合は、まずは馴染みのあるものから:
Java コード
new Thread() { public void run() { System.out.println("Thread running!"); }; }.start();
実際、上記のコードは、Thread のサブクラスを宣言し、Thread の run() メソッドをオーバーライドしてから、サブクラスのインスタンスを作成し、その start() メソッドを呼び出すことを意味します。 Thread の宣言されたサブクラスには名前がないため、匿名クラスと呼ばれます。名前のないクラスはクラスまたはメソッド内にのみ存在できるため、匿名内部クラスとも呼ばれます。
匿名内部クラスの構文は次のように書くこともできます:
Java コード
Thread thread = new Thread() { public void run() { System.out.println("Thread running!"); }; }; thread.start();
唯一の違いは、サブクラスを直接作成してそのメソッドを呼び出す代わりに、サブクラスの親クラス参照スレッドを宣言することです。親クラスの参照が子クラスのメソッドを呼び出します。
匿名クラスのインスタンス作成後、すぐにはstart()が実行されず、インスタンス作成とインスタンス実行のメソッドが分離されています。
この 2 つの違いは次のとおりです:
Java コード
//1 new User().setName("Boyce Zhang"); //2 User user = new User(); user.setName("Boyce Zhang");
匿名内部クラスの別の構文シナリオ:
Java コード
new Thread() { public void run() { System.out.println("Thread running!"); }; { start(); } };
実際、この書き方は、次のクラス ローカル コード ブロックにあります。匿名サブクラス そのクラスメソッドを呼び出します。
ローカル コード ブロック内のステートメントは、クラスのインスタンスが作成された直後にクラス ローダーによって暗黙的に実行されます。
同等:
Java コード
public class MyThread extends Thread { { start(); } public void run() { System.out.println("Thread running!"); }; }
したがって、3 つのメソッド間の実行時間のわずかな違いを除けば、効果には大きな違いはありません。
このように、Map を初期化する前述の方法は理解するのが難しくありません:
Java コード
private final static Map<String, String> CONSTANT = new HashMap<String, String>() { { put("1", "one"); put("2", "two"); } };
原理は次のとおりです:
HashMap のサブクラスを宣言してインスタンス化します (サブクラスは親クラスのメソッドをオーバーライドしません) HashMap ) を呼び出し、サブクラスのクラス ローカル コード ブロックで親クラス HashMap の put() メソッドを呼び出します。
最後に、インスタンス化された HashMap サブクラスのインスタンスを指す Map インターフェイス参照 CONSTANT を宣言します。
前の例によれば、HashMap の匿名サブクラスがインスタンス化された後、クラス ローカル コード ブロック内の put() メソッド呼び出しがクラス ローダーによって暗黙的に実行されることがわかります。
実際、Java のどのクラスやインターフェイスでも、それを継承または実装する匿名クラスを宣言できます。例:
Java コード
//重写父类方法,局部代码块调用自己重写过的父类方法。 List<String> list = new ArrayList<String>() { public boolean add(String e) { System.out.println("Cannot add anything!"); } //代码块的顺序在前后都无所谓,可以出现在类范围的任何位置。 { add("Boyce Zhang"); } }; //局部代码块调用父类方法。 dao.add(new User(){ { setName("Boyce Zhang"); setAge(26); } }); //重写父类方法 ThreadLocal<User> threadLocal = new ThreadLocal<User>() { protected String initialValue() { return new User("Boyce Zhang", 26); } };
匿名クラス内では、その親クラスのメソッドを実装またはオーバーライドするだけではありません。
また、独自のメソッドまたはそのクラスのローカル コード ブロックでその親クラスのメソッドを実行することもできます。
これは匿名内部クラス用の特別な構文ではなく、どのクラスにも適用される Java 構文です。
この書き方は、クラスをインスタンス化した直後に特定のメソッドを実行して、一部のクラス インスタンスのデータを初期化するためによく使用されます。
その機能は、最初にクラスをインスタンス化し、次にその参照を使用して、すぐに呼び出す必要があるメソッドを呼び出すことと同じです。
Java コード
Map<String, String> map = new HashMap<String, String>(); map.put("1", "one"); map.put("2", "two");
この構文の利点は、シンプルであることです。クラスをインスタンス化した直後に何かを実行する 物事がさらに便利になります。
この効果は、JavaScript のインスタント関数に似ています。しかし、本質的な違いがあります。
JavaScript にはクラスの概念がないため、つまり、JavaScript の関数はクラスであり、クラスは関数であるため、インスタント関数はロード後に関数全体を実行します。 Java のローカル コード ブロックは、クラスの任意のメソッドを実行することを選択できます。
もちろん、この書き方には欠点もあります。
内部クラスの各インスタンスは、外部クラスへの参照を暗黙的に保持します (静的内部クラスを除く) 一方で、これは冗長な参照の無駄です。一方、このサブクラス インスタンスをシリアル化するときに、外部クラスがシリアル化インターフェイスを実装していない場合は、エラーが報告されます。
Java の匿名内部クラスに関連するその他の記事については、PHP 中国語 Web サイトに注目してください。

ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

AI Hentai Generator
AIヘンタイを無料で生成します。

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

SublimeText3 中国語版
中国語版、とても使いやすい

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

ホットトピック









匿名の内部クラスはメモリ リークを引き起こす可能性があります。問題は、匿名の内部クラスが外部クラスへの参照を保持し、外部クラスのガベージ コレクションが妨げられることです。解決策には次のものが含まれます。 1. 弱参照を使用します。外部クラスが強参照によって保持されなくなった場合、ガベージ コレクターは弱い参照オブジェクトをすぐにリサイクルします。ガベージ コレクション中にメモリが必要になるため、ソフト参照オブジェクトがリサイクルされます。 Android アプリケーションなどの実戦では、匿名内部クラスによるメモリ リークの問題は、弱参照を使用することで解決でき、リスナーが不要な場合には匿名内部クラスを再利用できます。

匿名内部クラスは、明示的な名前を持たず、new 式を通じて作成される Java の特別な内部クラスであり、主に特定のインターフェイスの実装または抽象クラスの拡張に使用され、作成直後に使用されます。一般的な匿名内部クラスの設計パターンには次のものがあります。 アダプター パターン: 1 つのインターフェイスを別のインターフェイスに変換します。戦略パターン: アルゴリズムの定義と置換。オブザーバー パターン: オブザーバーを登録し、イベントを処理します。これは、文字列の長さによる TreeSet の並べ替え、匿名スレッドの作成など、実際のアプリケーションで非常に役立ちます。

匿名内部クラスは、サブクラス化、コードの簡素化、およびイベント (ボタンのクリックなど) の処理を容易にする特別な内部クラスとして Java で使用されます。実際のケースは次のとおりです。 イベント処理: 匿名の内部クラスを使用して、ボタンのクリック イベント リスナーを追加します。データ変換: Collections.sort メソッドと匿名内部クラスをコンパレータとして使用してコレクションを並べ替えます。

匿名内部クラスの使用エラー: 非スレッドセーフ環境で未宣言の例外をキャッチすることを使用してスコープ外の変数にアクセスする

匿名内部クラスのパフォーマンスの問題は、それらが使用されるたびに再作成されることです。これは、次の戦略によって最適化できます。 1. 匿名内部クラスをローカル変数に格納する。 2. 非静的内部クラスを使用する。 3. ラムダを使用する。表現。実際のテストでは、ラムダ式の最適化が最も効果的であることが示されています。

匿名内部クラスは、プライベート メンバーにアクセスする必要がある、複数のインスタンスが必要な、継承が必要な、ジェネリック型にアクセスする必要がある場合の使用には適していません。

匿名内部クラスの有効期間は、そのスコープによって決まります。 メソッドローカル内部クラス: それを作成したメソッドのスコープ内でのみ有効です。コンストラクターの内部クラス: 外部クラスのインスタンスにバインドされ、外部クラスのインスタンスが解放されると解放されます。静的内部クラス: 外部クラスと同時にロードおよびアンロードされます。

匿名の内部クラスにより、マルチスレッド コードの作成が簡素化され、名前を付ける必要がなくなり、スレッド クラスの即時定義と使用が可能になります。主な利点はコードを簡素化できることですが、コードを拡張できないという制限があります。 1 つまたは 2 つのスレッドをすばやく作成する必要がある場合に使用します。より複雑なロジックが必要な場合は、別のクラス ファイルを作成する必要があります。
