Java中的Kotlin 内部类原理

作者:??自动化BUG制造器???? 时间:2021-10-26 23:02:13 

Java 中的内部类

这是一个 Java 内部类的简单实现:

public class OutterJava {
   private void printOut() {
       System.out.println("AAA");
   }

   class InnJava {
       public void printInn() {
           printOut();
       }
   }
}

外部类是一个私有方法,内部类为什么可以访问到外部类的私有方法呢?思考这个问题,首先要从它的字节码入手,看看 JVM 到底对 java 文件做了什么。

字节码分析流程是:

  • javac xxx.java生成 class 文件。

  • javap -c xxx.class对代码进行反汇编,可以生成可查看的代码内容。

通过 javac 命令生成 class 文件,此时会发现生成了两个 class 文件,一个外部类 OtterJava 的,一个内部类 InnJava 的。

OutterJava.class

OutterJava.class 反汇编后的代码如下所示,这里面除了一个构造方法,多生成了一个

Compiled from "OutterJava.java"
public class java.OutterJava {
 public java.OutterJava();
   Code:
      0: aload_0
      1: invokespecial #2                  // Method java/lang/Object."<init>":()V
      4: return

 private void printOut();
   Code:
      0: getstatic     #3                  // Field java/lang/System.out:Ljava/io/PrintStream;
      3: ldc           #4                  // String AAA
      5: invokevirtual #5                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
      8: return

 static void access$000(java.OutterJava);
   Code:
      0: aload_0
      1: invokespecial #1                  // Method printOut:()V
      4: return
}

从反编译出来的内容来看,多了一个静态的access$000(OutterJava)方法,它的内部调用了 printOut()

InnJava.class

Compiled from "OutterJava.java"
class java.OutterJava$InnJava {
 final java.OutterJava this$0;

 java.OutterJava$InnJava(java.OutterJava);
   Code:
      0: aload_0
      1: aload_1
      2: putfield      #1                  // Field this$0:Ljava/OutterJava;
      5: aload_0
      6: invokespecial #2                  // Method java/lang/Object."<init>":()V
      9: return

 public void printInn2();
   Code:
      0: aload_0
      1: getfield      #1                  // Field this$0:Ljava/OutterJava;
      4: invokestatic  #3                  // Method java/OutterJava.access$000:(Ljava/OutterJava;)V
      7: return
}

在 InnJava 的字节码反编译出来的内容中,主要有两个点需要注意:

  • 构造方法需要一个外部类参数,并把这个外部类实例保存到了this$0中。

  • 调用外部类私有方法,实际上是调用了OutterJava.access$000方法。

小结:

在 Java 中,内部类与外部类的关系是:

  • 内部类持有外部类的引用,作为内部构造参数传入外部类实例,并保存到了内部类的属性this$0中。

  • 内部类调用外部类的私有方法,实际上是外部类生成了内部实际调用私有方法的静态方法access$000,内部类可以通过这个静态方法访问到外部类中的私有方法。

Kotlin 中的内部类

同样的 Java 代码,用 Kotlin 实现:

class Outter {
   private fun printOut() {
       println("Out")
   }

   inner class Inner {
       fun printIn() {
           printOut()
       }
   }
}

这里如果不加inner关键字,printIn()内的printOut()会报错Unresolved reference: printOut 。

不加inner关键字,反编译后的字节码:

public final class java/Outter$Inner {
 // ...
 public <init>()V
  L0
   LINENUMBER 8 L0
   ALOAD 0
   INVOKESPECIAL java/lang/Object.<init> ()V
   RETURN
  L1
   LOCALVARIABLE this Ljava/Outter$Inner; L0 L1 0
   MAXSTACK = 1
   MAXLOCALS = 1
 // ...
}

不加inner关键字,内部类的构造方法是没有外部类实例参数的。如果加上inner,就和 Java 一样:

// 加上了 inner 的构造方法
 public <init>(Ljava/Outter;)V
  L0
   LINENUMBER 8 L0
   ALOAD 0
   ALOAD 1
   PUTFIELD java/Outter$Inner.this$0 : Ljava/Outter;
   ALOAD 0
   INVOKESPECIAL java/lang/Object.<init> ()V
   RETURN
  L1
   LOCALVARIABLE this Ljava/Outter$Inner; L0 L1 0
   LOCALVARIABLE this$0 Ljava/Outter; L0 L1 1
   MAXSTACK = 2
   MAXLOCALS = 2

而内部类对于外部类私有方法的访问,也是通过静态方法access$XXX来实现的:

public final static synthetic access$printOut(Ljava/Outter;)V
  L0
   LINENUMBER 3 L0
   ALOAD 0
   INVOKESPECIAL java/Outter.printOut ()V
   RETURN
  L1
   LOCALVARIABLE $this Ljava/Outter; L0 L1 0
   MAXSTACK = 1
   MAXLOCALS = 1

来源:https://juejin.cn/post/7090789271866441735

标签:Java,Kotlin,内部类
0
投稿

猜你喜欢

  • java之Object类用法实例

    2023-11-05 04:14:26
  • Spring中事务几个常见的问题解决

    2022-04-07 00:52:01
  • IntelliJ IDEA 如何配置git的操作方法

    2021-12-28 11:24:44
  • Android开发之使用SQLite存储数据的方法分析

    2022-09-17 12:13:12
  • Spring Cloud中FeignClient实现文件上传功能

    2023-06-23 07:57:09
  • Android开发使用Drawable绘制圆角与圆形图案功能示例

    2023-08-26 21:07:09
  • Java实现读取文章中重复出现的中文字符串

    2022-04-27 04:29:05
  • Android 自定义一套 Dialog通用提示框 (代码库)

    2022-10-08 06:20:31
  • Android视频悬浮窗口实现的示例代码

    2022-08-01 06:50:33
  • SpringBoot与Angular2的集成示例

    2021-09-02 22:24:55
  • Android studio 3.5.2安装图文教程详解

    2022-06-27 19:15:46
  • 用Java实现24点游戏

    2022-07-18 20:56:14
  • Spring Boot缓存实战之Redis 设置有效时间和自动刷新缓存功能(时间支持在配置文件中配置)

    2023-11-11 01:57:18
  • Android应用中仿今日头条App制作ViewPager指示器

    2023-02-01 15:45:24
  • C# DataTable.Select()根据条件筛选数据问题

    2021-10-14 16:53:34
  • 用Rational Rose逆向工程(java)生成类图(教程和错误解决)

    2023-05-20 20:34:14
  • Java通过SSM完成水果商城批发平台流程

    2023-01-04 01:32:04
  • 基于ElasticSearch Analyzer的使用规则详解

    2023-09-28 14:41:04
  • Java抽象类和接口的区别详情

    2023-05-23 20:09:59
  • Spring 应用中集成 Apache Shiro的方法

    2023-02-18 05:44:55
  • asp之家 软件编程 m.aspxhome.com