6内部类
注意:内部类是一个编译时的概念,一旦编译成功,就会成为完全不同的两类。对于一个名为outer的外部类和其内部定义的名为inner的内部类。编译完成后出现outer.class和outer$inner.class两类。所以内部类的成员变量/方法名可以和外部类的相同。
6.1成员内部类
public class Outer { public class Inner { public void print(String str) { System.out.println(str); } } public Inner getInner() { return new Inner(); } public static void main(String[] args) { //因为成员内部类需要先创建了外部类对象,才能创建内部类 Outer outer = new Outer(); //1.外部类对象.new 内部类() Outer.Inner inner = outer.new Inner(); inner.print("Outer.new"); //2.使用getInner()来获取成员内部类,尤其是该内部类的构造函数无参数时(推荐) inner = outer.getInner(); inner.print("Outer.get"); } }
6.2局部内部类
public class Parcel4 { public Destination destination(String s) { class PDestination implements Destination { private String label; private PDestination(String whereTo) { label = whereTo; } public String readLabel() { return label; } } return new PDestination(s); } public static void main(String[] args) { Parcel4 p = new Parcel4(); Destination d = p.destination("Tasmania"); } }
定义在作用域里:
public class Parcel5 { private void internalTracking(boolean b) { if (b) { class TrackingSlip { private String id; TrackingSlip(String s) { id = s; } String getSlip() { return id; } } TrackingSlip ts = new TrackingSlip("slip"); String s = ts.getSlip(); } } public void track() { internalTracking(true); } public static void main(String[] args) { Parcel5 p = new Parcel5(); p.track(); } }
6.3嵌套内部类
public class Outer { public static class Inner { public Inner(){ System.out.println("执行Inner()构造方法"); System.out.println("b=外部类.内部类.内部类成员"); } public static void print(String str) { System.out.println(str); } public static String a="外部类.内部类.内部类静态成员"; public String b="外部类.内部类.内部类成员"; } public static void main(String[] args) { //外部类.内部类.内部类静态方法 Outer.Inner.print("外部类.内部类.内部类静态方法"); //外部类.内部类.内部类静态成员 System.out.println(Outer.Inner.a); // System.out.println(Outer.Inner.b);//报错,显示必须为静态成员变量 Inner inner1=new Outer.Inner(); //执行了静态类内部类的构造器 //结果:执行Inner()构造方法 // b=外部类.内部类.内部类成员 } }
如果一个类要被声明为static的,只有一种情况,就是静态内部类。如果在外部类声明为static,程序会编译都不会过。在一番调查后个人总结出了3点关于内部类和静态内部类(俗称:内嵌类)
1.静态内部类跟静态方法一样,只能访问静态的成员变量和方法,不能访问非静态的方法和属性,但是普通内部类可以访问任意外部类的成员变量和方法。
2.静态内部类可以声明普通成员变量和方法,而普通内部类不能声明static成员变量和方法。
3.静态内部类可以单独初始化:
Inner i = new Outer.Inner();
普通内部类初始化:
Outer o = new Outer(); Inner i = o.new Inner();
静态内部类使用场景一般是当外部类需要使用内部类,而内部类无需外部类资源,并且内部类可以单独创建的时候会考虑采用静态内部类的设计。
6.4 匿名内部类:Anonymous inner class
6.4.1匿名内部类概念
-
- 使用匿名内部类的前提:
内部类可以继承或实现一个外部类或者接口
-
- 什么情况下,内部类只被使用一次呢?
最常见的就是方法的形参列表上
-
- 多态+实现接口
-
- 多态+继承抽象类
6.4.2注意事项
- 使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口。
- 匿名内部类中是不能定义构造函数的。
- 匿名内部类中不能存在任何的静态成员变量和静态方法。
- 匿名内部类为局部内部类,所以局部内部类的所有限制同样对匿名内部类生效。
- 匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。
6.4.3外部类的方法的形参被final修饰
public class Outer { public static void main(String[] args) { Outer outer = new Outer(); Inner inner = outer.getInner("Inner", "gz"); System.out.println(inner.getName()); } public Inner getInner(final String name, String city) { return new Inner() { private String nameStr = name; public String getName() { return nameStr; } }; } } //注释后,编译时提示类Inner找不到 /* interface Inner { String getName(); } */
6.4.5匿名类有带参数的构造函数
public class Outer { public static void main(String[] args) { Outer outer = new Outer(); Inner inner = outer.getInner("Inner", "gz"); System.out.println(inner.getName()); } public Inner getInner(final String name, String city) { return new Inner(name, city) { private String nameStr = name; public String getName() { return nameStr; } }; } } abstract class Inner { Inner(String name, String city) { System.out.println(city); } abstract String getName(); }
note:抽象类带有带参构造方法
继承子类子类需要重写带参构造方法:
6.4.5匿名内部类通过实例初始化,可以达到类似构造器的效果
public class Outer { public static void main(String[] args) { Outer outer = new Outer(); Inner inner = outer.getInner("Inner", "gz"); System.out.println(inner.getName()); System.out.println(inner.getProvince()); } public Inner getInner(final String name, final String city) { return new Inner() { private String nameStr = name; private String province; // 实例初始化 { if (city.equals("gz")) { province = "gd"; }else { province = ""; } } public String getName() { return nameStr; } public String getProvince() { return province; } }; } } interface Inner { String getName(); String getProvince(); }
6.5内部类的继承
首先在继承语句extends处,注意命名空间,需要加上外围类名,才能对应上正确的内部类。
其次是构造对象上,这里需要自己写一个接受外围类引用的构造器,来给导出类提供创建对象的基本环境。
注意在构造器中的这一句wi.super()这是必须在构造器中使用的,才能够成功的构造出一个继承自内部类的对象。及enclosingClassReference.super()这个语法一定要牢记。
public class InheritInner extends Outer.Inner { // InheritInner() 是不能通过编译的,一定要加上形参 InheritInner(Outer wi) { wi.super("1"); } public static void main(String[] args) { Outer wi = new Outer(); InheritInner obj = new InheritInner(wi); } } class Outer { public Outer(){ System.out.println("构造Outer()..."); } class Inner { public Inner(){ System.out.println("构造Inner()..."); } public Inner(String name){ System.out.println("构造Inner(name)..."); } } }
class A{ public A(){ System.out.println("A()...构造方法"); } public A(String name){ System.out.printf("A(s%)...构造方法",name); } } class B extends A { public B(){ System.out.println("B()...构造方法"); } public B(String name){ System.out.printf("B(s%)...构造方法",name); } } public class ExtendsDemo { public static void main(String[] args) { A b=new B(); }
可是此处确实不是这样。
内部类会被覆盖吗?
到一个外部类继承自另一个含有内部类的父类。然后在该类中重写了父类的内部类,这个时候会怎么样呢?父类的内部类会被覆盖吗?
public class Egg { private Yolk y; protected class Yolk{ public Yolk() { System.out.println("Egg.Yolk!"); } } public Egg(){ System.out.println("New Egg"); y = new Yolk(); } }
public class BigEgg extends Egg { public class Yolk{ public Yolk() { System.out.println("BigEgg yolk"); } } public static void main(String[] args) { new BigEgg(); } }
重写的内部类并没有被调用,说明了在不同的外围类中的内部类是相互独立的实体,他们存在于自己的命名空间中,如果想要实现覆盖的话,可以直接使用继承语法,将子类的内部类继承自父类的内部类
public class Egg1 { protected class Yolk{ public Yolk() { System.out.println("Egg1.Yolk"); } public void f() { System.out.println("Egg1.Yolk.f()"); } } public Egg1() { System.out.println("new Egg1()"); } private Yolk y = new Yolk(); public void insertYolk(Yolk yy) { y = yy; } public void g() { y.f(); } }
public class BigEgg1 extends Egg1{ public class Yolk extends Egg1.Yolk{ public Yolk() { System.out.println("BigEgg1.Yolk()"); } public void f() { System.out.println("BigEgg1.Yolk.f()"); } } public BigEgg1() { insertYolk(new Yolk()); } public static void main(String[] args) { Egg1 e1 = new BigEgg1(); e1.g(); } }
来源:https://www.cnblogs.com/wqbin/p/11172942.html