匿名内部类是什么
首先, 如果在一个A类里面定义一个B类, 那么B类就是内部类, A类是外部类. 内部类就相当于外部类的一个成员, 你可以把内部类看成一个整体. 它又分为静态内部类和非静态内部类.
匿名内部类是非静态内部类的一种特殊情况, 匿名即没有类名, 因此就不可能有构造函数, 不能创建对象.
但请注意实质上匿名内部类是有类名和构造函数的. 由编译器创建.
为什么会有匿名内部类
为了程序员的方便以及代码的简洁.本来程序员应该先创建一个类继承抽象类或实现接口再创建对象, 但很多情况下, 这个类只会被使用一次, 似乎单独存在的必要性不大.. 为了简便, 匿名内部类允许我们在主方法当中进行抽象类/接口的实例化, 同时也可以进行对象的创建. 比如移动端开发最常见的 View.onClickListener
这个接口:
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// TO DO
}
}
直接通过匿名内部类的方式, 在主方法中实现接口创建一个匿名对象传递给按钮. 除了简洁以外, 还有其他的作用, 在下文单独讲解作用.
定义
匿名内部类的两种定义方式:
new 实现接口()
{
// 匿名内部类类体部分
}
new 父类构造器(实参列表)
{
// 匿名内部类类体部分
}
这两种方式的定义分别对应两种方式,一种是接口,另一种是父类构造器.
对于实现接口, 由于接口是没有构造函数的, 注意这里一定是空参数.
第二种是调用父类的构造器, 注意此处可以是空参数, 也可以传入参数.
作用
匿名内部类除了代码简洁以外, 主要有以下几个用途:
- 覆盖父类的方法
- 实现接口的方法
- 使用匿名内部类传入代码块进行初始化
覆盖父类的方法
覆盖父类的 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;
这个测试条件在用于匿名子类时会失效!