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
