美文网首页
Java内部类使用

Java内部类使用

作者: Codes作业本 | 来源:发表于2020-10-05 14:23 被阅读0次

    内部类的种类与相关属性

    1. 成员内部类

      含义:是普通的内部类,定义在一个类的内部

      属性:

      • 内部类访问外部类的属性或方法

        • 可以访问外部类的所有成员属性和成员方法(包括private成员和静态成员)

          //代码省略
          
        • 当内部类和外部类的变量或方法重名时,默认访问的是成员内部类的属性和方法

          //如果要访问外部类的属性或方法
          外部类.this.成员变量
          外部类.this.成员方法
          
      • 外部类访问内部属性或方法

        • 外部类必须创建一个成员内部类的对象,再通过此内部类对象的引用来访问成员内部类属性或方法

          public class A{
              public static void main(String args[]){
                  B b = new B();
                  b.getB();
              }
              class B{
                  public void getB(){
                      System.out.println("print getB()");
                  }
              }
          }
          
     - 成员内部类需要依附外部类而存在,如果C类想要使用A的内部类B,则必须创建一个外部类的对象
    
       ```java
       public class C{
        public static void main(String args[]){
               //第一种
            A a = new A();
               A.B b1 = a.new B();
               b.getB();
               //第二种(仅表达可以用单例方式获取)
               A.B b2 = a.getInstance();
        }
       }
       public class A{
           public B getInstance(){
               return new B();
           }
        class B{
            public getB(){
                System.out.println("print getB()");
               }
        }
       }
       ```
    
    1. 局部内部类

      含义:定义在一个方法或者一个作用域里面的类

      属性:

      局部内部类与方法中的局部变量一样,不能有作用域(public,protected,private)及static

      仅限于方法内或者改作用域内

      //代码省略
      
    2. 匿名内部类

      含义:定义一个类的同事对其进行实例化,与局部类相似,不同的是没有复制给某个变量

      属性:

      匿名内部类也是不能有访问修饰符(public,protected,private)和static修饰符的。

      匿名内部类可以访问外部类内所有成员

      匿名内部类是唯一一种没有构造器的类。正因为其没有构造器,所以匿名内部类的使用范围非常有限,大部分匿名内部类用于接口回调。匿名内部类在编译的时候由系统自动起名为Outter$1.class(1为整数)。一般来说,匿名内部类用于继承其他类或是实现接口,并不需要增加额外的方法,只是对继承方法的实现或是重写。

      匿名内部类不能访问外部类未加final修饰的变量(注意:JDK1.8即使没有用final修饰也可以访问)下文有具体解释

      • 衍生:匿名方法,匿名对象

        匿名函数:详见lambda表达式?

        匿名对象含义:即创建对象时,只有创建对象的语句,却没有把对象地址值赋值给某个变量。

        由于没有指定引用变量,所以只能使用一次,每次使用都会创建

        //创建一个普通对象
        A a = new A();
        //创建一个匿名对象
        new A();
        
    1. 静态内部类

      含义:在一个类的里面定义一个通过static修饰的内部类

      属性:

      • 静态内部类调用外部类

        可以访问外部类静态成员变量或方法,并且作用域private也可

      • 外部类调用静态内部类

        外部类可以直接调用静态内部类

        public class C{
            public static void main(String args[]){
                //调用内部类的静态方法
                A.B.getMethodeB1();
                //调用内部类的非静态方法
                //间接访问需要创建内部类对象
                A.B b = new A.B();
                b.getMethodB2();
            }
        }
        class A{
            static class B{
                public static void getMethodB1(){
                   System.out.println("print getMethodB1()");
                }
                public void getMethodB2(){
                   System.out.println("print getMethodB2()");
                }
            }
        }
        

    内部类作用与好处

    1. 可以无条件的访问外部类的所有元素

    2. 方便将存在一定逻辑关系的类组织在一起,又可以实现对外隐藏

    3. 可以实现多重继承

      每个内部类都能独立的集成一个接口的实现,所以无论外部类是否已经继承某个接口的实现,对于内部都没有影响。使得多继承的额解决方案变得完整。

    4. 通过匿名内部类来优化简单的接口实现

    内部类可能引入的问题

    内部类的使用可能造成程序的内存泄漏:

    如果一个匿名内部类没有被任何引用持有,那么匿名内部类对象用完就有机会被回收。

    因为内部类持有指向外部类的引用,就会造成GC无法回收内部类或者外部类

    匿名内部类引入问题

    为什么匿名内部类访问局部变量必须要用final修饰??

    匿名内部类之所以可以访问局部变量,是因为在底层将这个局部变量的值传入到了匿名内部类中,并且以匿名内部类的成员变量的形式存在,这个值的传递过程是通过匿名内部类的构造器完成的。

    final的修饰是为了保证数据的一致性:

    对引用变量来说是引用地址的一致性,对基本类型来说就是值的一致性

     public class Hello$1 extends Thread {
        
        private String val$str;
        
        Hello$1(String paramString) {
            this.val$str = paramString;
        }
     
        public void run() {
            System.out.println(this.val$str);
        }
     
    }
    

    final修饰符对变量来说,深层次的理解就是保障变量值的一致性。为什么这么说呢?因为引用类型变量其本质是存入的是一个引用地址,说白了还是一个值(可以理解为内存中的地址值)。用final修饰后,这个这个引用变量的地址值不能改变,所以这个引用变量就无法再指向其它对象了。

    回到正题,为什么需要用final保护数据的一致性呢?

    因为将数据拷贝完成后,如果不用final修饰,则原先的局部变量可以发生变化。这里到了问题的核心了,如果局部变量发生变化后,匿名内部类是不知道的(因为他只是拷贝了局部变量的值,并不是直接使用的局部变量)。这里举个栗子:原先局部变量指向的是对象A,在创建匿名内部类后,匿名内部类中的成员变量也指向A对象。但过了一段时间局部变量的值指向另外一个B对象,但此时匿名内部类中还是指向原先的A对象。那么程序再接着运行下去,可能就会导致程序运行的结果与预期不同。

    现在我们来谈一谈JDK8对这一问题的新的知识点。 在JDK8中如果我们在匿名内部类中需要访问局部变量,那么这个局部变量不需要用final修饰符修饰。 看似是一种编译机制的改变,实际上就是一个语法糖(底层还是帮你加了final)。但通过反编译没有看到底层为我们加上final,但我们无法改变这个局部变量的引用值,如果改变就会编译报错。

    参考:

    https://www.cnblogs.com/dolphin0520/p/3811445.html

    https://blog.csdn.net/tianjindong0804/article/details/81710268

    相关文章

      网友评论

          本文标题:Java内部类使用

          本文链接:https://www.haomeiwen.com/subject/nadauktx.html