匿名内部类

2019-10-07  本文已影响0人  逗比喵喵

匿名内部类是什么

首先, 如果在一个A类里面定义一个B类, 那么B类就是内部类, A类是外部类. 内部类就相当于外部类的一个成员, 你可以把内部类看成一个整体. 它又分为静态内部类非静态内部类.

匿名内部类是非静态内部类的一种特殊情况, 匿名即没有类名, 因此就不可能有构造函数, 不能创建对象.

但请注意实质上匿名内部类是有类名和构造函数的. 由编译器创建.

为什么会有匿名内部类

为了程序员的方便以及代码的简洁.本来程序员应该先创建一个类继承抽象类或实现接口再创建对象, 但很多情况下, 这个类只会被使用一次, 似乎单独存在的必要性不大.. 为了简便, 匿名内部类允许我们在主方法当中进行抽象类/接口的实例化, 同时也可以进行对象的创建. 比如移动端开发最常见的 View.onClickListener 这个接口:

button.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
        // TO DO
    }
}

直接通过匿名内部类的方式, 在主方法中实现接口创建一个匿名对象传递给按钮. 除了简洁以外, 还有其他的作用, 在下文单独讲解作用.

定义

匿名内部类的两种定义方式:

new 实现接口()
{
    // 匿名内部类类体部分
}

new 父类构造器(实参列表)
{
    // 匿名内部类类体部分
}

这两种方式的定义分别对应两种方式,一种是接口,另一种是父类构造器.

对于实现接口, 由于接口是没有构造函数的, 注意这里一定是空参数.
第二种是调用父类的构造器, 注意此处可以是空参数, 也可以传入参数.

作用

匿名内部类除了代码简洁以外, 主要有以下几个用途:

  1. 覆盖父类的方法
  2. 实现接口的方法
  3. 使用匿名内部类传入代码块进行初始化

覆盖父类的方法

覆盖父类的 run() 方法:

new Thread() {
    @Override
    public void run() {
        // super.run();
        // TO DO
    }
}

实现接口的方法

即前面提到的 View.OnClickListener 接口:

button.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
        // TO DO
    }
}

使用匿名内部类传入代码块进行初始化

假设我们有代码:

List<String> friends = new ArrayList<>();
friends.add("Harry");
friends.add("Tony");
invite(friends);

借助双括号初始化 (double brace initialization) 技巧, 若这个 friends 数组之后不会再使用的话, 我们可以把它构造成一个匿名列表:

invite(new ArrayList<String>() {
    {
        add("Harry");
        add("Tony");
    }
});

匿名内部类的名字

特殊的一点是, 匿名, 当然是没有名字, 但其实匿名内部类是有名字的. 只是不是人类认知意义上的名字, 无法在其他地方使用:

Foo foo = new Foo() {
    @Override
    int bar() {
        return 0;
    }
}

这个内部类, 在字节码文件 .class 中也会被定义出来.

class package.name.outerClassName$1. 

内部类的命名方式为外部类名 + $ + N, N 是匿名内部类的顺序. 从 1 开始.

注意事项

建立一个与超类类似, 但不完全相同的匿名子类非常容易, 但这样的子类对象在使用 equals 方法时要特别当心, 我们在定义 equals 时, 一般要对类型进行测试:

if(getClass() != other.getClass()) return false;

这个测试条件在用于匿名子类时会失效!

参考

上一篇下一篇

猜你喜欢

热点阅读