在说匿名类之前,我想先说一下匿名对象的概念
匿名对象:没有名字的对象
匿名对象的应用场景:
1、调用方法的使用,仅仅是调用一次的时候
注意:调用多次的时候,使用匿名对象不合适
那么,为什么还会存在匿名对象呢?
匿名对象在调用完毕之后,由于栈内存中没有引用指向它,那么调用完后,就是一个垃圾空间,可以被垃圾回收器回收
2、匿名对象当作参数传递
class Cat{
public void eat(){
System.out.println("猫吃鱼");
}
}
class CatFactory{
//当一个类引用数据类型作为参数进行传递的时候,实际上传递的是该类实例化对象的地址值
public void makeCat(Cat c){ //Cat c = new Cat();
c.eat();
}
}
public class AnonymousDemo {
public static void main(String[] args) {
Cat c = new Cat();
// //匿名对象 没有名字的对象
// new Cat().eat();
// new Cat().eat();
//这里每一次new都相当于在堆内存中创建一个新的对象
CatFactory cf = new CatFactory();
//匿名对象当作参数传递
//下面这两种写法都是正确的
cf.makeCat(new Cat());
cf.makeCat(c);
// 甚至于像下面这样写也是正确的
// new CatFactory().makeCat(new Cat());
}
}
在介绍匿名内部类之前,先来说一下内部类
内部类就是将一个类定义在某个类或者方法里面,一共分为四种:成员内部类、局部内部类、匿名内部类、静态内部类
匿名内部类:就是内部类的简化写法
前提:存在一个类或者接口,这里的类可以是具体类也可以是抽象类
定义匿名内部类的语法格式如下
new 类名或者接口名() {重写方法;}
然后还要再提一下多态的概念
多态体现为父类引用变量可以指向子类对象。
下面来看一个例子
首先定义接口,这个接口定义了两个抽象方法,分别是奔跑run()和获取奔跑速度的getSpeed()
package test.AnonymousDemo2;
public interface Run {
public void run();
public int getSpeed();
}
接下来,先来看一下匿名内部类长什么样子
package test.AnonymousDemo2;
public class Anonymous2 {
public void personRun(Run r){
r.run();
System.out.println("the man's running speed is "+r.getSpeed());
}
public static void main(String[] args) {
Anonymous2 a = new Anonymous2();
a.personRun(new Run() {
@Override
public void run() {
System.out.println("can run");
}
@Override
public int getSpeed() {
return 3;
}
});
}
}
匿名内部类就是这个,实现了接口中的两个抽象方法
new Run() {
@Override
public void run() {
System.out.println("can run");
}
@Override
public int getSpeed() {
return 3;
}
}
最后的输出结果为
实际上,上面的匿名类可以写成
package test.AnonymousDemo2;
public interface Run {
public void run();
public int getSpeed();
}
package test.AnonymousDemo2;
public class Run1 implements Run {
public void run() {
System.out.println("can run");
}
public int getSpeed() {
return 3;
}
}
package test.AnonymousDemo2;
public class Anonymous2 {
public void personRun(Run r){
r.run();
System.out.println("the man's running speed is "+r.getSpeed());
}
public static void main(String[] args) {
Anonymous2 a = new Anonymous2();
Run1 r = new Run1();
a.personRun(r);
//这里也可以直接写成
//a.personRun(new Run1());
}
}
结果为
两个代码的结果是一样的,其实匿名内部类相当于一个对象
文章的前面提过到过多态,父类的引用变量指向子类对象。这里的Run r 就可以理解为引用变量,然后匿名内部类就会作为实现接口的实例化子类对象,向上转型为Run类型,将对象传入方法personRun中,所以 r 可以调用 run 方法
如果还不清楚的话,还可以看下面这个例子
package test.AnonymousDemo2;
public interface Run {
public void run();
public int getSpeed();
}
package test.AnonymousDemo2;
public class Run1{
public void test(){
new Run(){
public void run() {
System.out.println("can run");
}
public int getSpeed() {
return 3;
}
};
}
}
package test.AnonymousDemo2;
public class Anonymous2 {
public static void main(String[] args) {
Run1 r1 = new Run1();
r1.test();
}
}
结果什么都不会打印,因为这里的
只是一个对象,什么方法都没有调用,所以什么结果都不会输出
现在将代码修改为
package test.AnonymousDemo2;
public class Run1{
public void test(){
new Run(){
public void run() {
System.out.println("can run");
}
public int getSpeed() {
return 3;
}
}.run();
}
}
再重新运行main函数
成功调用了 run() 方法
相关文章
暂无评论...