匿名内部类

成员内部类

格式

修饰符 class 外部类名称 {
    修饰符 class 内部类名称 {
        ...
    }
    ...
}

访问权限

  • 内部类使用外部类内容,可以随意访问
  • 外部类访问内部类内容,需要内部类对象

使用方式

间接方式

在外部类是方法当中,使用内部类,然后main方法中只是调用外部类的方法即可

package cn.itcast.day11.demo03;

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();
    }
    // get set方法
}

间接的去调用

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

直接方式

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

直接调用

public static void main(String[] args) {
        Body.Heart heart = new Body().new Heart();
        heart.beat();
    }

内部类的同名的变量的访问

如果出现同名的情形,那么调用格式如下:

外部类名称.this.外部类成员变量名

示例如下:

public class Outer {
    int num = 10; // 外部类的成员变量
    public class Inner /*extends Object*/ {
        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 MyOuter {
    public void methodOuter() {
        int num = 10; // 所在方法的局部变量
        class MyInner {
            public void methodInner() {
                System.out.println(num);
            }
        }
        // 使用
        MyInner inner = new MyInner();
        inner.methodInner();
    }
}

外部调用

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

局部内部类的权限修饰符

  • 最外层的类可以使用 pulic / (default)这两种修饰符来修饰
  • 成员内部类可以使用publicprotected(default)private来修饰
  • 局部内部类,不能有修饰符(即使是default也不行)

局部内部类访问方法所在局部变量

如果局部内部类想要访问放在方法的局部变量,那么这个局部变量就必须是【有效final的】

  • jdk8之前(不包含jdk8)必须要final关键字
  • jdk8及以后可以不用final关键字,只要符合只赋值一次也行。

有效final的解释

  • 可以使用定义局部变量的时候使用final关键字
  • 如果没有使用final关键字,但是有与使用了final后一样的效果也是可以的,也就是变量定义后只有一次赋值机会。

原理分析

至于为什么局部内部类如果想要使用所在方法的局部变量必须是final的与局部变量和对象的生命周期有关。

  • new 出来的对象是在堆内存中。
  • 局部变量是跟着方法走,也就是在栈内存中(方法需要进栈才可以运行)。
  • 方法运行结束后,立刻出栈,局部变量就会立刻消失。
  • new出来的对象会在堆内存中持续存在,直到垃圾回收消失。

经过上面的分析之后可以直到,若方法中运行完了,那么局部变量也就消失了,但是局部内部类的对象还会存在,若在局部内部类中使用了局部变量,就会报错,所以需要将其定义为final的,这样局部内部类会将局部变量的值拷贝一份出来,这样即使局部变量的声明周期已经结束也没有关系。

匿名内部类

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

格式

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

  • new代表创建对象的动作
  • 接口名称就是匿名内部类需要实现哪个接口
  • {...}这才是匿名内部类的内容

示例

接口

public interface MyInterface {
    void method1();
    void method2();
}

使用匿名内部类

public class DemoMain {
    public static void main(String[] args) {
        // 使用匿名内部类,但不是匿名对象,对象名称就叫objA
        MyInterface objA = new MyInterface() {
            @Override
            public void method1() {
                System.out.println("匿名内部类实现了方法!111-A");
            }

            @Override
            public void method2() {
                System.out.println("匿名内部类实现了方法!222-A");
            }
        };
        objA.method1();
        objA.method2();
}

注意事项

匿名内部类

在【创建对象】的时候,只能使用唯一一次。如果希望多次创建对象,而且类的内容一样的话,那么就需要使用单独定义的实现类了。

匿名对象

在【调用方法】的时候,只能调用唯一一次。如果希望同一个对象,调用多次方法,那么必须给对象起个名字。

匿名内部类是省略了【实现类/子类名称】,但是匿名对象是省略了【对象名称】

public class DemoMain {
    public static void main(String[] args) {
        // 使用匿名内部类,但不是匿名对象,对象名称就叫objA
        MyInterface objA = new MyInterface() {
            @Override
            public void method1() {
                System.out.println("匿名内部类实现了方法!111-A");
            }

            @Override
            public void method2() {
                System.out.println("匿名内部类实现了方法!222-A");
            }
        };
        objA.method1();
        objA.method2();
        System.out.println("=================");

        // 使用了匿名内部类,而且省略了对象名称,也是匿名对象
        new MyInterface() {
            @Override
            public void method1() {
                System.out.println("匿名内部类实现了方法!111-B");
            }

            @Override
            public void method2() {
                System.out.println("匿名内部类实现了方法!222-B");
            }
        }.method1();
        // 因为匿名对象无法调用第二次方法,所以需要再创建一个匿名内部类的匿名对象
        new MyInterface() {
            @Override
            public void method1() {
                System.out.println("匿名内部类实现了方法!111-B");
            }

            @Override
            public void method2() {
                System.out.println("匿名内部类实现了方法!222-B");
            }
        }.method2();
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值