Java内部类

本文详细介绍了Java内部类的四种类型:成员内部类、局部内部类、静态内部类和匿名内部类,以及它们各自的访问规则和特性。内部类可以提供更好的封装性和实现多继承的效果,同时在回调功能和解决方法重名问题上也有独特作用。文章还举例说明了内部类在实际编程中的应用场景,如实现回调功能和解决接口继承冲突。
摘要由CSDN通过智能技术生成

参考链接:https://www.zhihu.com/question/26954130/answer/708467570      https://www.cnblogs.com/dolphin0520/p/3811445.html

(一) 概述

把类定义在另一个类的内部,该类就被称为内部类。

 

(二) 内部类的访问规则

​ A:可以直接访问外部类的成员,包括私有

​ B:外部类要想访问内部类成员,必须创建对象

(三) 内部类的分类

​ A:成员内部类

​ B:局部内部类

​ C:静态内部类

​ D:匿名内部类

(1) 成员内部类

成员内部类——就是位于外部类成员位置的类
特点:可以使用外部类中所有的成员变量和成员方法(包括private的)

成员内部类常见修饰符:

A:private

如果我们的内部类不想轻易被任何人访问,可以选择使用private修饰内部类,这样我们就无法通过创建对象的方法来访问,想要访问只需要在外部类中定义一个public修饰的方法,间接调用。这样做的好处就是,我们可以在这个public方法中增加一些判断语句,起到数据安全的作用。

  class Outer {
      private class Inner {
          public void show() {
              System.out.println(“密码备份文件”);
          }
      }
      
      public void method() {
          if(你是管理员){
              Inner i = new Inner();
              i.show();
          }else {
              System.out.println(“你没有权限访问”);
          }
      }
  }

下面我们给出一个更加规范的写法

  class Outer {
      private class Inner {
          public void show() {
              System.out.println(“密码备份文件”);
          }
      }
      //使用getXxx()获取成员内部类,可以增加校验语句(文中省略)
      public Inner getInner() {
          return new Inner();
      }
      
      public static void main(String[] args) {
          Outer outer = new Outer();
          Outer.Inner inner = outer.getInner();
          inner.show();
      }
  }

 

(2) 局部内部类

局部内部类——就是定义在一个方法或者一个作用域里面的类
特点:主要是作用域发生了变化,只能在自身所在方法和属性中被使用

访问时:

  //在局部位置,可以创建内部类对象,通过对象调用和内部类方法
  class Outer {
      private int age = 20;
      public void method() {
          final int age2 = 30;
          class Inner {
              public void show() {
                  System.out.println(age);
                  //从内部类中访问方法内变量age2,需要将变量声明为最终类型。
                  System.out.println(age2);
              }
          }
          
          Inner i = new Inner();
          i.show();
      }
  }

 

(3) 静态内部类

我们所知道static是不能用来修饰类的(static目的为了修饰的成员能随着类的加载而加载,用来修饰外部类无意义),但是成员内部类可以看做外部类中的一个成员,所以可以用static修饰,这种用static修饰的内部类我们称作静态内部类,也称作嵌套内部类.
特点:不能使用外部类的非static成员变量和成员方法

解释:非静态内部类编译后会默认的保存一个指向外部类的引用,而静态内部类却没有。

简单理解

即使没有外部类对象,也可以创建静态内部类对象,而外部类的非static成员必须依赖于对象的调用,静态成员则可以直接使用类调用,不必依赖于外部类的对象,所以静态内部类只能访问静态的外部属性和方法。

(4) 匿名内部类

一个没有名字的类,是内部类的简化写法。本质:其实是继承该类或者实现接口的子类匿名对象

A 格式:

  new 类名或者接口名() {
      重写方法();
  }

这也就是下例中,可以直接使用 new Inner() {}.show(); 的原因 == 子类对象.show();

  interface Inner {
      public abstract void show();
  }
  ​
  class Outer {
      public void method(){
          new Inner() {
              public void show() {//我是重写的父类Inner的抽象方法
                  System.out.println("HelloWorld");
              }
          }.show();//用来调用对象的方法去运行。
      }
  }
  ​
  class Test {
      public static void main(String[] args)  {
          Outer o = new Outer();
          o.method();
      }
  }    

 

 

B:匿名内部类在开发中的使用

​我们在开发的时候,会看到抽象类,或者接口作为参数。

而这个时候,实际需要的是一个子类对象。

如果该方法仅仅调用一次,我们就可以使用匿名内部类的格式简化。

 

C:匿名内部类是有名字的,由JVM运行时生成的名字,命名规则:外部类+$+内部类的个数

 

 

 

D:编译器定义匿名内部类的构造方法。

如下图,Client的匿名内部类的构造方法为Client$1(Client client,OuterClass outerClass).默认传入自己的外部类实例和自己的非静态父类的外部实例。其目的在初始化内部类时,需要表明自己的作用域是在父类为outclass实例作用在外部类实例为client中的。

 

接口和静态内部类效果一样都是静态的,不需要使用外部类的实例,所以此处只需要自己外部类的实例。

 

如果run方法也是静态方法,那么静态方法不需要外部类的实例,此时一个外部类的实例都不需要了。

局部内部类/匿名内部类访问局部变量必须加final修饰。外部作用域的局部变量object,如果内部类要引用此变量,此变量必须final。

 

为什么局部内部类和匿名内部类只能访问局部final变量?

1、因为局部变量是随着方法的调用而调用使用完毕就消失而堆内存的数据并不会立即消失。所以,堆内存还是用该变量,而该变量已经没有了。为了让该值还存在,就加final修饰。

2、通过上文讲解我们可知,在上面这段代码中编译时会生成两个class文件:Client.class和Client$1.class,Client$1.class其实就是内部类的class文件,此时在编译时,Client$1.class会拷贝Client.class中的局部变量object,即捕获外部变量,此时Client.class和Client$1.class中的object就是完全独立的两个值了,如果此时在test改变了object就会导致数据不一致性,这样就达不到原本的意图和要求。为了解决这个问题,java编译器就限定必须将变量object限制为final变量,不允许指向新的对象(对于基本类型的变量,是不允许对变量进行更改),这样数据不一致性的问题就得以解决了。

(即上文(2)局部内部类中的例子: age2 的位置存储着常量30 而不是 age2 这个变量名)

 

SAM类型:“SAM”代表“单一抽象方法”,“SAM类型”是指像Runnable,Callable等接口.Lambda表达式,Java 8中的一个新功能,被认为是SAM类型,可以自由转换为它们。

只能是拥有一个方法的接口,抽象类和多方法的接口都不行。

匿名内部类都有哪些限制?


 

使用内部类的原因

(一) 封装性

作为一个类的编写者,我们很显然需要对这个类的使用访问者的访问权限做出一定的限制,我们需要将一些我们不愿意让别人看到的操作隐藏起来,

如果我们的内部类不想轻易被任何人访问,可以选择使用private修饰内部类,这样我们就无法通过创建对象的方法来访问,想要访问只需要在外部类中定义一个public修饰的方法,间接调用。

比如上面提到的“密码备份文件”的代码。

这样做的好处之一就是,我们可以在这个public方法中增加一些判断语句,起到数据安全的作用。

其次呢,我们的对外可见的只是getInner()这个方法,它返回了一个Demo接口的一个实例,而我们真正的内部类的名称就被隐藏起来了

(二) 实现多继承 ※

我们之前的学习知道,java是不可以实现多继承的,一次只能继承一个类,我们学习接口的时候,有提到可以用接口来实现多继承的效果,即一个接口有多个实现,但是这里也是有一点弊端的,那就是,一旦实现一个接口就必须实现里面的所有方法,有时候就会出现一些累赘,但是使用内部类可以很好的解决这些问题

  public class Demo1 {
      public String name() {
          return "BWH_Steven";
      }
  }
  
  public class Demo2 {
      public String email() {
          return "xxx.@163.com";
      }
  }
  
  public class MyDemo {
  ​
      private class test1 extends Demo1 {
          public String name() {
              return super.name();
          }
      }
  ​
      private class test2 extends Demo2  {
          public String email() {
              return super.email();
          }
      }
  ​
      public String name() {
          return new test1().name();
      }
  ​
      public String email() {
          return new test2().email();
      }
  ​
      public static void main(String args[]) {
          MyDemo md = new MyDemo();
          System.out.println("我的姓名:" + md.name());
          System.out.println("我的邮箱:" + md.email());
      }
  }

我们编写了两个待继承的类Demo1和Demo2,在MyDemo类中书写了两个内部类,test1和test2 两者分别继承了Demo1和Demo2类,这样MyDemo中就间接的实现了多继承

(三) 用匿名内部类实现回调功能

我们用通俗讲解就是说在Java中,通常就是编写一个接口,然后你来实现这个接口,然后把这个接口的一个对象作以参数的形式传到另一个程序方法中, 然后通过接口调用你的方法,匿名内部类就可以很好的展现了这一种回调功能

  public interface Demo {
      void demoMethod();
  }
  
  public class MyDemo{
      public test(Demo demo){
          System.out.println("test method");
      }
      
      public static void main(String[] args) {
          MyDemo md = new MyDemo();
          //这里我们使用匿名内部类的方式将接口对象作为参数传递到test方法中去了
          md.test(new Demo){
              public void demoMethod(){
                  System.out.println("具体实现接口")
              }
          }
      }
  }

(四) 解决继承及实现接口出现同名方法的问题

编写一个接口 Demo

  public interface Demo {
      void test();
  }

编写一个类 MyDemo

  public class MyDemo {
  ​
      public void test() {
          System.out.println("父类的test方法");
      }
      
  }

编写一个测试类

  public class DemoTest extends MyDemo implements Demo {
      public void test() {
      }
  }

这样的话我就有点懵了,这样如何区分这个方法是接口的还是继承的,所以我们使用内部类解决这个问题

  public class DemoTest extends MyDemo {
  ​
  ​
      private class inner implements Demo {
          public void test() {
              System.out.println("接口的test方法");
          }
      }
      
      public Demo getIn() {
          return new inner();
      }
      
      
      public static void main(String[] args) {
          //调用接口而来的test()方法
          DemoTest dt = new DemoTest();
          Demo d = dt.getIn();
          d.test();
          
          //调用继承而来的test()方法
          dt.test();
      }
  }
  ​
  //运行结果
  接口的test方法
  父类的test方法

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值