zoukankan      html  css  js  c++  java
  • 匿名内部类详解

    内部类(理解)
    (1)把类定义在另一个类的内部,该类就被称为内部类。
    举例:把类B定义在类A中,类B就被称为内部类。
    (2)内部类的访问规则
    A:可以直接访问外部类的成员,包括私有
    B:外部类要想访问内部类成员,必须创建对象
    (3)内部类的分类
    A:成员内部类
    B:局部内部类
    (4)成员内部类
    A:private 为了数据的安全性
    B:static 为了访问的方便性

    成员内部类不是静态的:
    外部类名.内部类名 对象名 = new 外部类名.new 内部类名();
    成员内部类是静态的:
    外部类名.内部类名 对象名 = new 外部类名.内部类名();
    (5)成员内部类的面试题(填空)
    30,20,10

    class Outer {
    public int num = 10;

    class Inner {
    public int num = 20;

    public viod show() {
    int num = 30;

    System.out.println(num);
    System.out.println(this.num);
    System.out.println(Outer.this.num);
    }
    }
    }
    (6)局部内部类
    A:局部内部类访问局部变量必须加final修饰。
    B:为什么呢?
    因为局部变量使用完毕就消失,而堆内存的数据并不会立即消失。
    所以,堆内存还是用该变量,而改变量已经没有了。
    为了让该值还存在,就加final修饰。
    通过反编译工具我们看到了,加入final后,堆内存直接存储的是值,而不是变量名。
    (7)匿名内部类(掌握)
    A:是局部内部类的简化形式
    B:前提
    存在一个类或者接口
    C:格式:
    new 类名或者接口名() {
    重写方法;
    }
    D:本质:
    其实是继承该类或者实现接口的子类匿名对象
    (8)匿名内部类在开发中的使用
    我们在开发的时候,会看到抽象类,或者接口作为参数。
    而这个时候,我们知道实际需要的是一个子类对象。
    如果该方法仅仅调用一次,我们就可以使用匿名内部类的格式简化。

    interface Person {
    public abstract void study();
    }

    class PersonDemo {
    public void method(Person p) {
    p.study();
    }
    }

    class PersonTest {
    public static void main(String[] args) {
    PersonDemo pd = new PersonDemo();
    pd.method(new Person() {
    public void study() {
    System.out.println("好好学习,天天向上");
    }
    });
    }
    }

    (9)匿名内部类的面试题(补齐代码)
    interface Inter {
    void show();
    }

    class Outer {
    //补齐代码
    public static Inter method() {
    return new Inter() {
    public void show() {
    System.out.println("HelloWorld");
    }
    };
    }
    }

    class OuterDemo {
    public static void main(String[] args) {
    Outer.method().show(); //"HelloWorld"
    }
    }

    匿名内部类也就是没有名字的内部类

    正因为没有名字,所以匿名内部类只能使用一次,它通常用来简化代码编写

    但使用匿名内部类还有个前提条件:必须继承一个父类或实现一个接口

    实例1:不使用匿名内部类来实现抽象方法

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    abstract class Person {
        public abstract void eat();
    }
     
    class Child extends Person {
        public void eat() {
            System.out.println("eat something");
        }
    }
     
    public class Demo {
        public static void main(String[] args) {
            Person p = new Child();
            p.eat();
        }
    }

    运行结果:eat something

    可以看到,我们用Child继承了Person类,然后实现了Child的一个实例,将其向上转型为Person类的引用

    但是,如果此处的Child类只使用一次,那么将其编写为独立的一个类岂不是很麻烦?

    这个时候就引入了匿名内部类

    实例2:匿名内部类的基本实现

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    abstract class Person {
        public abstract void eat();
    }
     
    public class Demo {
        public static void main(String[] args) {
            Person p = new Person() {
                public void eat() {
                    System.out.println("eat something");
                }
            };
            p.eat();
        }
    }

    运行结果:eat something

    可以看到,我们直接将抽象类Person中的方法在大括号中实现了

    这样便可以省略一个类的书写

    并且,匿名内部类还能用于接口上

     

    实例3:在接口上使用匿名内部类

    interface Person {
        public void eat();
    }
     
    public class Demo {
        public static void main(String[] args) {
            Person p = new Person() {
                public void eat() {
                    System.out.println("eat something");
                }
            };
            p.eat();
        }
    }

    运行结果:eat something

    由上面的例子可以看出,只要一个类是抽象的或是一个接口,那么其子类中的方法都可以使用匿名内部类来实现

    最常用的情况就是在多线程的实现上,因为要实现多线程必须继承Thread类或是继承Runnable接口

    实例4:Thread类的匿名内部类实现

    public class Demo {
        public static void main(String[] args) {
            Thread t = new Thread() {
                public void run() {
                    for (int i = 1; i <= 5; i++) {
                        System.out.print(i + " ");
                    }
                }
            };
            t.start();
        }
    }

    运行结果:1 2 3 4 5

    实例5:Runnable接口的匿名内部类实现

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    public class Demo {
        public static void main(String[] args) {
            Runnable r = new Runnable() {
                public void run() {
                    for (int i = 1; i <= 5; i++) {
                        System.out.print(i + " ");
                    }
                }
            };
            Thread t = new Thread(r);
            t.start();
        }
    }

    运行结果:1 2 3 4 5

  • 相关阅读:
    linux 安装 svn
    人群计数:Single-Image Crowd Counting via Multi-Column Convolutional Neural Network
    Ubuntu查看CPU占用和使用情况
    看完这篇文章,我奶奶都懂了HTTPS原理
    EfficientNet算法笔记
    Soft NMS算法笔记
    DenseNet算法详解
    AI竞赛服务平台—— FlyAI
    Cornernet训练自己的数据
    深度学习物体检测:CornerNet
  • 原文地址:https://www.cnblogs.com/dandandeyoushangnan/p/4653783.html
Copyright © 2011-2022 走看看