【Java】内部类
嵌套类(Nested Class)
入れ子クラス(Nested Class)
staticメンバークラス
内部クラス(Inner Class)
非staticメンバークラス
匿名クラス
静态成员类
-
- 特定クラス(MyClass)にクラス(MyHelper)が依存、MyHelperはMyClassからしか呼ばれない
-
- →1個の.javaファイルにまとめることを考える
トップレベルクラス:ネストしてないファイル直下のクラス
エンクロージングクラス:入れ子のクラスを内側に含んだクラス
親クラスX:エンクロージングクラスかつトップレベルクラス
子クラスY:ZのエンクロージングクラスでXの入れ子クラス
孫クラスZ:Yの入れ子クラス
public class MyClass{...}
class MyHelper{...}
-
- しかし、MyHelperはあくまでパッケージプライベートで、同じパッケージからは自由にアクセスできる
-
- →staticでMyClassのプライベートにすべき!
エンクロージングクラスのclassブロック配下で入れ子クラス定義すると
入れ子クラスはエンクロージングクラス外からは見えない
入れ子クラスの名前はエンクロージングクラス.入れ子クラス(MyClass.MyHelper)
エンクロージング内部ではメンバークラス名で呼び出し可能
import (package名).MyClass.MyHelper;でエンクロージング外からもvar h = new MyHelper();可能
//classブロック配下でMyHelper定義
public class MyClass {
//staticメンバークラスの定義
//メンバーはMyClass外からは見えない
private static class MyHelper {
public void show() {
System.out.println("Nested class is running!");
}
}
public void run() {
//エンクロージング内部ではMyHelper呼び出し可能
var helper = new MyHelper();
helper.show();
}
}
public class NestBasic {
public static void main(String[] args) {
var c = new MyClass();
c.run();
//MyHelpeはMyClass外からは見えない
// var h = new MyClass.MyHelper(); //エラー!
}
}
非静态成员类(=内部类)
-
- static修飾子なし
内部クラス:エンクロージングオブジェクトに所属
staticメンバークラス:エンクロージングクラスに所属
内部クラスの特徴
内部クラスをインスタンス化するにはエンクロージングオブジェクト必要
内部クラスからはthis変数経由でエンクロージングクラスメンバーにアクセス
エンクロージングクラスのインスタンスフィールド参照用途に使用
個々のインスタンスがエンクロージングオブジェクトへの参照を持つのでメモリを消費する
class MyClass {
private String str1 = "エンクロージング・インスタンス";
private static String str2 = "エンクロージング・クラス";
private class MyHelper {
private String str1 = "ネスト・インスタンス";
private static final String str2 = "ネスト・クラス";
public void show() {
//内部クラスはエンクロージングオブジェクトへの参照を持つ
//MyClass.thisでアクセス可
System.out.println(MyClass.this.str1); //エンクロージング・インスタンス
System.out.println(MyClass.str2); //エンクロージング・クラス
}
}
public void run() {
//内部クラスのインスタンスはエンクロージングクラスのインスタンスメソッドで生成
var helper = new MyHelper();
helper.show();
//エンクロージングオブジェクトから内部クラスにアクセス
System.out.println(helper.str1); //ネスト・インスタンス
System.out.println(MyHelper.str2); //ネスト・クラス
}
}
public class NestedAccess {
public static void main(String[] args) {
var c = new MyClass();
c.run();
}
}
非静态成员类(内部类)的实例
-
- private内部クラスを宣言したらエンクロージング外からは見えない
- 見えなくなるのは型名のみ、インスタンスの参照は制限なし
package java.util;
import java.util.function.Consumer;
public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {
//...中略
/*Iterators
Itrクラスの内部的な実装を見せる必要はない
外部からは実装元であるIterators型が見えたらいい
iteratorメソッドの戻り値はIterator型
*/
public Iterator<E> iterator() {
return new Itr();
}
//...中略
/*内部クラス
リスト内のprivateフィイールドにアクセス、配下要素を順にアクセスするイテレータを実装
イテレータはIterator実装クラスとして別に定義する
*/
private class Itr implements Iterator<E> {
//...中略
}
//...中略
}
匿名类(无名类)
特定の処理をまとめて表すための器として使う
ex:イベントリスナー
特定の文脈(ユーザーの動作など)に強く関連するので他では再利用しない
new 基底クラス(引数,…){クラスの本体}
new View.OnClickListener(){
匿名クラスには名前がないのでnew演算子に基底クラス/インターフェースの名前を指定
用意したリスナーのインスタンスにはそのままメソッドに渡せる
注意;
コンストラクタを持てない(初期化はOK)
継承不可(暗黙的にfinal)
抽象クラス宣言不可
//イベントリスナー登録
btn.setOnClickListener(
//匿名クラス(イベントリスナー)定義
new View.OnClickListener(){
//btnクリックでtxtに現在時刻を表示
@Override
public void onClick(View view){
TextView txt = findViewById(R.id.txtResult);
txt.setText(LocalDateTime.now().toString());
}
}
);
本地类
-
- メソッド、コンストラクタなどブロック配下で定義されたクラス
クラスが特定のメソッド、コンストラクタで利用されるときに使う
匿名クラスで代用OK