「javaの内部クラス」、
意外と知ってそうで知らない箇所が多かったので、おさらいしてみようと。
内部クラスとは
言葉の通り、クラス内部のクラスである。
内部クラスの用途
用途としては、もっと良い隠蔽(カプセル化)を提供する。
外部クラスを離れては意味のないクラスを内部に隠蔽することができる。
例:Cow.javaとCowLeg.java
内部クラスの分類
- 非static内部クラスは、__外部クラスのインスタンスに関連__する
- static内部クラスは、__外部クラスに関連に関連__する
- local内部クラスは、__メソッドに関連__する
どれに関連するかは非常に大事な内容であり、
それによって、使い方が決まっていたことが分かった。
非static内部クラスの特徴
外部クラスの非staticのフィールド、メソッド、コンストラクター等と同じ扱いである。
⇒ インスタンスが存在しないと使えないもの。
classでありながら、private,protectedが使える
(staticも付けれるが、static内部クラスになるので後で)
public class Outer {
public class PublicInner {}
private class PrivateInner {}
public class PackagePrivateInner {}
}
- classの上位単位はpackageであり、public、package privateで十分
- 非static内部クラスの上位単位は外部クラスなので、すべての可視性が使える
内部は外部が見えるが、外部は内部が見えない
public class Outer {
private int outerVal;
public class Inner {
public int innerVal;
private void method() {
// OK
this.innerVal = Outer.this.outerVal;
}
}
private void method() {
// NG
// this.outerVal = innerVal;
}
}
非static内部クラスは、__外部クラスのインスタンスに関連している__ので、
- 「非static内部クラスインスタンスが存在する = 外部クラスインスタンスが存在する」
⇒ 内部から外部が見える - 「外部クラスインスタンスが存在する キ 非static内部クラスインスタンスが存在する」
⇒ 外部から内部が見えない
※ 非static内部クラスを使い場合は、明示的なインスタンス生成が必要。
非static内部クラスのインスタンス生成
Outer.Inner inner = (new Outer()).new Inner();
- (new Outer()) -> 外部クラスのインスタンス
- 外部クラス.new Inner() -> 外部クラスのインスタンス内に非static内部クラスのインスタンス生成
非static内部クラスのインスタンスは外部クラスから丸見え
public class Outer {
private int outerVal;
public class Inner {
private int privateA = 1;
protected int protectedB = 2;
public int publicC = 3;
}
private void method() {
Inner inner = new Inner();
System.out.println(inner.publicC);
System.out.println(inner.protectedB);
System.out.println(inner.privateA);
}
}
public static void main(String[] args) {
Outer.Inner inner = new Outer().new Inner();
System.out.println(inner.publicC);
System.out.println(inner.protectedB);
// NG
// System.out.println(inner.privateA);
}
内部クラスのインスタンスは、
- 外部クラスからはprivateまで見える
- 他のクラスからは可視性が働く
Outer$Inner@461
Innerは独立しているわけではなくて、Outerの一部だと理解した。
よって、同じクラス内なので、InnerのPrivateもOuterで見える。
static内部クラスも同じだった。
もっといい解釈を求めてます!!!
staticなfield,method,blockは使えない
コンパイルエラー:
「Inner classes cannot have static declarations」
public class Outer2 {
public class Inner {
static { /** something **/ }
public static int staticValue;
public static void staticMethod() { /** something **/ }
}
}
非static内部クラスは、__外部クラスのインスタンスに関連している__ので、
- staticは、インスタンスではなくクラスに関連付けるためのものなので、
文法からみて矛盾しているので、コンパイルエラー
変数の優先順位
public class Outer {
String a = "outer-a";
String b = "outer-b";
String c ="outer-c";
public class Inner {
String a = "inner-a";
String b = "inner-b";
private void method() {
String a = "method-a";
System.out.println(a);
System.out.println(b);
System.out.println(c);
System.out.println(this.a);
System.out.println(Outer.this.a);
}
}
public static void main(String[] args) {
new Outer().new Inner().method();
}
}
method-a
inner-b
outer-c
inner-a
outer-a
非static内部クラスの優先順位は
- メソッド内local変数
- 内部クラスのインスタンス変数
- 外部クラスのインスタンス変数
this.変数, __Outer.this.変数__で変数範囲を指定できる。
このデバックした図から分かるね!!
static内部クラスの特徴
ぶっちゃけ、通常のクラスと変わりません!!
が、内部クラスとしても特徴はあります。
classでありながら、private,protectedが使える
非static内部クラスと同じく、上位はクラスであるので、private,protectedが使える。
static内部クラスのインスタンスは外部クラスから丸見え
非static内部クラスと同じく、外部クラスへ所属しているので。
static内部クラスのインスタンス生成
Outer.Inner inner = new Outer.Inner();
- new 外部クラス.staticな内部クラス()で生成できる
※ 外部クラスがパッケージっぽくなってますね。
local内部クラス
public static void main(String[] args) {
class Inner {
protected int a = 1;
}
class InnerSub extends Inner {
private int b = 2;
}
InnerSub sub = new InnerSub();
System.out.println(sub.a + sub.b);
}
- local変数にprivate,protectedが不要と同じく、可視性は使えない
- メソッドに関連しているので、staticは使えない
まとめ
- 内部クラスを使う際には、__「何(クラス、インスタンス、メソッド)と関連させたいのか」__を明確にするのが重要だ。
- 原則として内部クラスはstaticにすることが推奨されます。
主な理由は、staticでない内部クラスが暗黙的にエンクロージングクラス(トップレベルクラス)のインスタンスを参照するためです。(セキュリティ面??)「this$0」
From Effective Java 第2版 (The Java Series)