内部类

内部类

含义

如果一个事物的内部包含另一事物,那么就是一个类内部包含另一个类。

例如:身体和心脏的关系。又如:汽车和发动机的关系。

编译此类

img

得到编译文件结构
$代表内部类的格式

分类

成员内部类

定义格式

修饰符 class 外部类名称{

		修饰符 class 内部类名称{

				//...

		}

		//...

}

注意:内部类使用外部类,随意访问;外部类使用内部类,需要内部对象。

public class Body {//外部类
    public class Heart{//成员内部类

        //内部类的方法
        public void beat(){
            System.out.println("心脏跳动,蹦蹦蹦");
            System.out.println("我叫:"+ name);//正确写法
        }
    }

    //外部类的成员变量
    private String name;

    //外部类的方法
    public  void methodBody(){
        System.out.println("外部类的方法");
        new Heart().beat();
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

使用方式

1.间接方式:在外部类的方法当中,使用内部类,然后main方法调用外部类的方法。

直接方式:公式:

外部类名称.内部类名称 对象名 = new 外部类名称().new 内部类名称

public class InnerClass {
    public static void main(String[] args) {
        Body body =new Body();//外部类的对象
        //通过外部类的对象,调用外部类的方法,里面间接使用内部类Heart。
        body.methodBody();
        System.out.println("=====================================");

        //按照公式写
        Body.Heart heart = new Body().new Heart();
        heart.beat();

    }
}

如何访问重名的外部类成员变量

public class Outer {
    int num =  10;//外部类的成员变量
    public class Inner{
        int num = 20;//内部类的成员变量
        public void methodInner(){
            int num = 30;//内部类方法的局部变量
            System.out.println(num);//局部变量,就近原则
            System.out.println(this.num);//内部类的成员变量
            System.out.println(Outer.this.num);//外部类的成员变量
        }
    }

}

局部内部类(包括匿名内部类)

含义

如果一个类定义在方法里,那么这就是一个局部内部类。
“局部”:只有当前所属的方法才能使用它,出了这个方法外面就不能用了。

定义格式

修饰符 class 外部类方法名称(参数列表){
class 局部内部类名称{
//...
}
}

public class Outer {
    public void methodOuter(){
        class   Inner{//局部内部类
            int num = 10;
            public void methodInner(){
                System.out.println(num);
            }
        }
    Inner inner = new Inner();
    inner.methodInner();
    }
}

使用方式

通过调用外部方法中自己定义的对象进行访问

public class DemoMain {
    public static void main(String[] args) {
        Outer obj = new Outer();
        obj.methodOuter();
    }
}

注意

局部内部类,如果希望访问所在方法的局部变量,那么这个局部变量就必须是有效final的

备注:从Java 8+开始,只要局部变量事实上不变,那么final关键字可以省略。

原因:
1.new 出来的对象在堆内存中。
2.局部变量是跟着方法走的,在栈内存当中。
3.方法运行结束后,立刻出栈,局部变量就回立刻消失。
4.但是new出来的对象在堆内存中持续存在,直到垃圾回收消失。
二者生命周期不同,内部类还存在的时候,局部变量可能已经消失,使用final保证它是一个常量。避免错误。

public class MyOuter {
    public void methodOuter(){
        int num = 10;//所在方法的局部变量
        class MyInner{
            public void methodInner(){
                System.out.println(num);
            }
        }
    }


}

小结:

外部类 public / (default)
成员内部类 public / protected / (default) / private
局部内部类 什么都不写,但是和(default)不一样,同一个包内可以访问(default),但是这个只有自己能用。

匿名内部类

含义

如果接口的实现类(或者父类的子类)只需要使用唯一的一次,那么这种情况下可以省略该类的定义,而改为匿名内部类

定义格式

接口名称 对象名 = new 接口名称(){
//覆盖重写所有抽象方法
};

对格式 “new 接口名称(){...}”进行解析
1.new 代表创建对象的动作
2.接口名称就是匿名内部类需要实现哪个接口
3.{...}这才是匿名内部类的内容

注意

1.匿名内部类,在创建对象的时候,只能使用唯一一次
如果希望多次创建对象,而且类的内容一样的话,那么就必须使用单独定义的接口实现类
2.匿名对象,在调用方法的时候,只能调用唯一一次,
如果希望同一个对象,调用多次方法,那么必须给对象起个名字
3.匿名内部类是省略了实现类/子类名称,但是匿名对象是省略了对象名称
二者不是一回事!!!

public class DemoMain {
    public static void main(String[] args) {
 //       MyInterface ipl = new MyInterfaceIpl();//多态
//       ipl.method();
//        MyInterface some = new MyInterface();//错误写法
        //使用匿名内部类,但不是匿名对象,对象名叫obj1
        MyInterface objA = new MyInterface() {
            @Override
            public void method1() {
                System.out.println("匿名内部类实现了方法!111A");
            }
            public void method2() {
                System.out.println("匿名内部类实现了方法!222A");
            }
        };
        objA.method1();
        objA.method2();
        System.out.println("===================================");
        //使用了匿名内部类,而且省略了对象名称,也是匿名对象
        /*MyInterface objB = */new MyInterface() {
            @Override
            public void method1() {
                System.out.println("匿名内部类实现了方法!111B");
            }
            @Override
            public void method2() {
                System.out.println("匿名内部类实现了方法!222B");
            }
        }.method1();
    //    objB.method1();
      //  objB.method2();
        new MyInterface() {
            @Override
            public void method1() {
                System.out.println("匿名内部类实现了方法!111B");
            }
            @Override
            public void method2() {
                System.out.println("匿名内部类实现了方法!222B");
            }
        }.method2();
    }
}
posted @ 2021-01-31 23:34  老虎彡  阅读(57)  评论(0)    收藏  举报