当前位置:   article > 正文

详解匿名内部类,看了都说好!

不同主构造器的匿名内部类

点击上方 "程序员小乐"关注, 星标或置顶一起成长

每天凌晨00点00分, 第一时间与你相约

每日英文

When one reaches a point of difficulty where there is no one else to rely on, s/he will often become stronger.

一个人若已到了没有任何东西可以依赖的时候,往往会变得更坚强。

每日掏心

人做任何事情都要看开一点。学会放弃,本身就是一种淘汰,一种选择,淘汰掉自己的弱项,选择自己的强项。

来自:chenssy | 责编:乐乐

链接:cnblogs.com/chenssy/

程序员小乐(ID:study_tech)第 712 次推文   图片来自网络

往日回顾:“高仿版拼多多”宣告破产!曾一年收割1.3亿用户,如今自救失败负债16亿

   正文   

          在这篇博客中你可以了解到匿名内部类的使用、匿名内部类要注意的事项、如何初始化匿名内部类、匿名内部类使用的形参为何要为final。

一、使用匿名内部类内部类

          匿名内部类由于没有名字,所以它的创建方式有点儿奇怪。创建格式如下:

  1. new 父类构造器(参数列表)|实现接口()      
  2. {       
  3.     //匿名内部类的类体部分      
  4. }

          在这里我们看到使用匿名内部类我们必须要继承一个父类或者实现一个接口,当然也仅能只继承一个父类或者实现一个接口。同时它也是没有class关键字,这是因为匿名内部类是直接使用new来生成一个对象的引用。当然这个引用是隐式的。

  1. public abstract class Bird {
  2.    private String name;
  3.    public String getName() {
  4.          return name;    
  5.    }    
  6.    public void setName(String name) {        
  7.        this.name = name;    
  8.    }        
  9.    public abstract int fly();
  10.    }
  11. public class Test {        
  12.       public void test(Bird bird){        
  13.            System.out.println(bird.getName() + "能够飞 "
  14.                 + bird.fly() + "米");
  15.       }        
  16.       public static void main(String[] args) {        
  17.           Test test = new Test();        
  18.           test.test(new Bird() {                        
  19.           public int fly() {                
  20.          return 10000;            
  21.       }                        
  22.       public String getName() {                
  23.          return "大雁";
  24.       }        
  25.     });    
  26.  }
  27. }
  28. ------------------Output: 大雁能够飞 10000

          在Test类中,test()方法接受一个Bird类型的参数,同时我们知道一个抽象类是没有办法直接new的,我们必须要先有实现类才能new出来它的实现类实例。所以在mian方法中直接使用匿名内部类来创建一个Bird实例。

         由于匿名内部类不能是抽象类,所以它必须要实现它的抽象父类或者接口里面所有的抽象方法。

          对于这段匿名内部类代码其实是可以拆分为如下形式:

  1. public class WildGoose extends Bird{
  2.    public int fly() {
  3.        return 10000;    
  4.    }        
  5.    public String getName() {
  6.        return "大雁";    
  7.    }
  8. }  
  9. WildGoose wildGoose = new WildGoose();
  10. test.test(wildGoose);

           在这里系统会创建一个继承自Bird类的匿名类的对象,该对象转型为对Bird类型的引用。

          对于匿名内部类的使用它是存在一个缺陷的,就是它仅能被使用一次,创建匿名内部类时它会立即创建一个该类的实例,该类的定义会立即消失,所以匿名内部类是不能够被重复使用。对于上面的实例,如果我们需要对test()方法里面内部类进行多次使用,建议重新定义类,而不是使用匿名内部类。

二、注意事项

      在使用匿名内部类的过程中,我们需要注意如下几点:

          1、使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口。

          2、匿名内部类中是不能定义构造函数的。

           3、匿名内部类中不能存在任何的静态成员变量和静态方法。

          4、匿名内部类为局部内部类,所以局部内部类的所有限制同样对匿名内部类生效。

           5、匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。

三、使用的形参为何要为final

参考链接:android.blog.51cto.com/268543/384844

           我们给匿名内部类传递参数的时候,若该形参在内部类中需要被使用,那么该形参必须要为final。也就是说:当所在的方法的形参需要被内部类里面使用时,该形参必须为final。

为什么必须要为final呢?

          首先我们知道在内部类编译成功后,它会产生一个class文件,该class文件与外部类并不是同一class文件,仅仅只保留对外部类的引用。当外部类传入的参数需要被内部类调用时,从java程序的角度来看是直接被调用:

  1. public class OuterClass {    
  2.    public void display(final String name,String age){        
  3.    class InnerClass{            
  4.        void display(){                
  5.            System.out.println(name);            
  6.        }        
  7.    }    
  8.    }
  9. }

      从上面代码中看好像name参数应该是被内部类直接调用?其实不然,在java编译之后实际的操作如下:

  1. public class OuterClass$InnerClass {    
  2.    public InnerClass(String name,String age){        
  3.        this.InnerClass$name = name;        
  4.        this.InnerClass$age = age;    
  5.    }              
  6.    public void display(){        
  7.        System.out.println(this.InnerClass$name +
  8.            "----" + this.InnerClass$age );    
  9.    }
  10. }

          所以从上面代码来看,内部类并不是直接调用方法传递的参数,而是利用自身的构造器对传入的参数进行备份,自己内部方法调用的实际上时自己的属性而不是外部方法传递进来的参数。

          直到这里还没有解释为什么是final?在内部类中的属性和外部方法的参数两者从外表上看是同一个东西,但实际上却不是,所以他们两者是可以任意变化的,也就是说在内部类中我对属性的改变并不会影响到外部的形参,而然这从程序员的角度来看这是不可行的,毕竟站在程序的角度来看这两个根本就是同一个,如果内部类该变了,而外部方法的形参却没有改变这是难以理解和不可接受的,所以为了保持参数的一致性,就规定使用final来避免形参的不改变。

          简单理解就是,拷贝引用,为了避免引用值发生改变,例如被外部类的方法修改等,而导致内部类得到的值不一致,于是用final来让该引用不可改变。

          故如果定义了一个匿名内部类,并且希望它使用一个其外部定义的参数,那么编译器会要求该参数引用是final的。

、匿名内部类初始化

        我们一般都是利用构造器来完成某个实例的初始化工作的,但是匿名内部类是没有构造器的!那怎么来初始化匿名内部类呢?使用构造代码块!利用构造代码块能够达到为匿名内部类创建一个构造器的效果

  1. public class OutClass {    
  2.    public InnerClass getInnerClass(final int age,final String name){
  3.        return new InnerClass() {            
  4.            int age_ ;            
  5.            String name_;                        
  6.            //构造代码块完成初始化工作          
  7.            {                
  8.                if(0 < age && age < 200){                    
  9.                age_ = age;                    
  10.                name_ = name;                
  11.            }            
  12.        }            
  13.        public String getName() {  
  14.             return name_;            
  15.        }                        
  16.        public int getAge() {              
  17.         return age_;            
  18.        }        
  19.     };    
  20.   }        
  21.   public static void main(String[] args) {
  22.          OutClass out = new OutClass();
  23.          InnerClass inner_1 = out.getInnerClass(201, "chenssy");
  24.          System.out.println(inner_1.getName());    
  25.          InnerClass inner_2 = out.getInnerClass(23, "chenssy");
  26.          System.out.println(inner_2.getName());    
  27.   }
  28. }

欢迎在留言区留下你的观点,一起讨论提高。如果今天的文章让你有新的启发,学习能力的提升上有新的认识,欢迎转发分享给更多人。

欢迎各位读者加入程序员小乐技术群,在公众号后台回复“加群”或者“学习”即可。

猜你还想看

阿里、腾讯、百度、华为、京东最新面试题汇集

Java 设计模式之责任链模式实现的三种方式

SpringBoot 常用注解和原理都在这儿了!

一文详解 Java 的八大基本类型!

关注「程序员小乐」,收看更多精彩内容

嘿,你在看吗

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小小林熬夜学编程/article/detail/683185
推荐阅读
相关标签
  

闽ICP备14008679号