內(nèi)部類應(yīng)用場(chǎng)景
描述事物A時(shí),發(fā)現(xiàn)需要在事物A內(nèi)部,封裝一個(gè)能訪問(wèn)A屬性等數(shù)據(jù)的事物B來(lái)只服務(wù)于事物A。
即如果一個(gè)類附屬于另一個(gè)類且只應(yīng)用于這個(gè)類,則可以用內(nèi)部類,比如Android的Adapter里的ViewHolder。
使用內(nèi)部類的優(yōu)勢(shì)
可以把內(nèi)部類隱藏在外部類內(nèi),不允許同一個(gè)包中的其他類訪問(wèn)該類。
內(nèi)部類定義于外部類的成員變量位置,因此內(nèi)部類可以使用任意訪問(wèn)控制符,如 public 、 protected 、 private 等,同時(shí)也可以訪問(wèn)外部類任意數(shù)據(jù)。
內(nèi)部類隱式的持有了一個(gè)外部類的引用,所能用外部類全部屬性,要想打破引用就要用靜態(tài)內(nèi)部類。
可以實(shí)現(xiàn)多重繼承。
成員內(nèi)部類
public static void main(String[] args) {
Outer.Inner inner = new Outer().new Inner();
inner.test();
}
class Outer {
public int i = 1;
class Inner {
public int i = 2;
public void test() {
System.out.println("避免就近原則,調(diào)用外部類同名變量" + Outer.this.i);
}
}
}
靜態(tài)內(nèi)部類
public static void main(String[] args) {
Outer.Inner inner = new Outer.Inner();//注意調(diào)用方式
inner.test();
}
class Outer {
public static int i = 1;
static class Inner {
public void test() {
System.out.println("靜態(tài)內(nèi)部類只能引用外部類的靜態(tài)成員" + i);
}
}
}
局部?jī)?nèi)部類
class Outer {
public void test() {
class Inner {
public void innerTest() {
System.out.println("InnerTest" );
}
}
Inner inner = new Inner();
inner.innerTest();
}
}
//調(diào)用內(nèi)部類方式
Outer outer = new Outer();
outer.test();
局部?jī)?nèi)部類訪問(wèn)局部變量問(wèn)題
class Outer {
int num = 3;
public void test() {
//JDK低于1.8
//final int x = 9;
//JDK1.8
int x = 9;
class Inner {
public void show() {
System.out.println("Inner...." + x);
}
}
Inner inner = new Inner();
inner.show();
}
}
如果你的JDK為1.8以下,想訪問(wèn)就要把局部變量聲明成final,而1.8后的JDK不需要(JDK1.8幫你隱藏了,實(shí)際還是存在final)。
至于為什么訪問(wèn)就要把局部變量聲明成fianl?
因?yàn)榫植孔兞康纳芷谑且坏┏鲎饔糜蚓婉R上被釋放,然而這時(shí)new Inner這個(gè)對(duì)象還在等待Java垃圾回收機(jī)制來(lái)回收并沒(méi)有馬上釋放消失,相應(yīng)的它內(nèi)部訪問(wèn)局部變量也沒(méi)有消失。
如此給人的感覺(jué)就是這個(gè)局部變量的生命周期延長(zhǎng)了,這就違背Java的設(shè)計(jì)原則了。
所以Sun想了解決方案,如果一個(gè)局部?jī)?nèi)部類訪問(wèn)一個(gè)局部變量的時(shí)候,那就讓該局部?jī)?nèi)部類訪問(wèn)該局部變量的復(fù)制品。
如此什么復(fù)制品?
完全一樣的才是復(fù)制品。
為了保證值完全一樣不能隨意變換,所以這個(gè)變量要聲明成final。
匿名內(nèi)部類
沒(méi)有類名的類就稱作匿名內(nèi)部類,只是沒(méi)有類名,類的其他成員都具備。
匿名內(nèi)部類的好處
簡(jiǎn)化書寫,能盡快的釋放空間(作為實(shí)參傳值,出作用域就釋放)。
繼承方式的匿名內(nèi)部類
abstract class Demo {
public abstract void read();
}
public class Run {
public static void main(String[] args) {
//注意這里創(chuàng)建的是繼承Demo類的子類對(duì)象
new Demo() {
@Override
public void read() {
System.out.println("Read");
}
}.read();
}
}
如果想通過(guò)匿名內(nèi)部類實(shí)現(xiàn)調(diào)用多個(gè)方法,需要將返回值設(shè)為基類
abstract class Demo {
public abstract Demo read();
public abstract Demo write();
}
public class Run {
public static void main(String[] args) {
// 注意這里是繼承Demo類的對(duì)象
new Demo() {
@Override
public Demo read() {
System.out.println("Read");
return this;
}
@Override
public Demo write() {
System.out.println("Write");
return this;
}
}.read().write();
}
}
如果匿名內(nèi)部類中定義了其特有的方法,想都不要想調(diào)用,不明白的去看我寫的多態(tài)篇找原因。
實(shí)現(xiàn)方式的匿名內(nèi)部類
//接口
interface Demo {
void read();
}
//測(cè)試類
class Test {
public void test() {
//注意:這是匿名的接口實(shí)現(xiàn)類對(duì)象,不是接口new。
new Demo() {
@Override
public void read() {
System.out.println("read");
}
}.read();
}
}
public class Run {
public static void main(String[] args) {
Test test = new Test();
test.test();
}
}
在實(shí)際開發(fā)中匿名內(nèi)部類一般用于實(shí)參
假設(shè)有一個(gè)函數(shù)的形參需要傳一個(gè)接口類的實(shí)參,這個(gè)實(shí)參僅在用時(shí)調(diào)用。
如果專門去寫一個(gè)實(shí)現(xiàn)類是不是太low,那么這時(shí)候就可以用上匿名內(nèi)部類。
interface Demo {
void read();
}
public class Run {
public static void main(String[] args) {
test(new Demo() {
@Override
public void read() {
System.out.println("Read");
}
});
}
public static void test(Demo demo) {
demo.read();
}
}