Java中的Kotlin 内部类原理


Posted in Java/Android onJune 16, 2022

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

总结

在 Kotlin 中,内部类持有外部类引用和通过静态方法访问外部类私有方法都是与 Java 一样的。唯一的不同是,Kotlin 中需要使用 inner关键字修饰内部类,才能访问外部类中的内容。实质是inner关键字会控制内部类的构造方法是否带有外部类实例参数。

到此这篇关于Java中的Kotlin 内部类原理的文章就介绍到这了,更多相关Java Kotlin 内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!


Tags in this post...

Java/Android 相关文章推荐
Java Shutdown Hook场景使用及源码分析
Jun 15 Java/Android
SpringBoot实现异步事件驱动的方法
Jun 28 Java/Android
Java基础之this关键字的使用
Jun 30 Java/Android
Java基于Dijkstra算法实现校园导游程序
Mar 17 Java/Android
Spring Boot DevTools 全局配置学习指南
Mar 31 Java/Android
SpringBoot中获取profile的方法详解
Apr 08 Java/Android
java中为什么说子类的构造方法默认访问的是父类的无参构造方法
Apr 13 Java/Android
java版 简单三子棋游戏
May 04 Java/Android
SpringBoot使用AOP实现统计全局接口访问次数详解
Jun 16 Java/Android
springboot读取resources下文件的方式详解
Jun 21 Java/Android
springboot创建的web项目整合Quartz框架的项目实践
Jun 21 Java/Android
SpringBoot项目多数据源及mybatis 驼峰失效的问题解决方法
Jul 07 Java/Android
Spring Security动态权限的实现方法详解
Java实现注册登录跳转
Jun 16 #Java/Android
Java界面编程实现界面跳转
springboot实现string转json json里面带数组
Jun 16 #Java/Android
Android Gradle 插件自定义Plugin实现注意事项
Jun 16 #Java/Android
Java完整实现记事本代码
Jun 16 #Java/Android
Springboot中如何自动转JSON输出
Jun 16 #Java/Android
You might like
Excel数据导入Mysql数据库的实现代码
2008/06/05 PHP
Display SQL Server Login Mode
2007/06/21 Javascript
javascript之通用简单的table选项卡实现(二)
2010/05/09 Javascript
jQuery EasyUI API 中文文档 - TreeGrid 树表格使用介绍
2011/11/21 Javascript
Javascript的getYear、getFullYear、getUTCFullYear异同分享
2011/11/30 Javascript
node.js中的buffer.copy方法使用说明
2014/12/14 Javascript
基于jQuery1.9版本如何判断浏览器版本类型
2016/01/12 Javascript
详解探索 vuex 2.0 以及使用 vuejs 2.0 + vuex 2.0 构建记事本应用
2017/06/16 Javascript
详解基于iview-ui的导航栏路径(面包屑)配置
2019/02/22 Javascript
vue 验证码界面实现点击后标灰并设置div按钮不可点击状态
2019/10/28 Javascript
node.js实现http服务器与浏览器之间的内容缓存操作示例
2020/02/11 Javascript
JS实现移动端可折叠导航菜单(现代都市风)
2020/07/07 Javascript
python遍历 truple list dictionary的几种方法总结
2016/09/11 Python
requests和lxml实现爬虫的方法
2017/06/11 Python
Python编程之变量赋值操作实例分析
2017/07/24 Python
Python实现在线暴力破解邮箱账号密码功能示例【测试可用】
2017/09/06 Python
pygame游戏之旅 如何制作游戏障碍
2018/11/20 Python
Python字符串通过'+'和join函数拼接新字符串的性能测试比较
2019/03/05 Python
计算机二级python学习教程(1) 教大家如何学习python
2019/05/16 Python
Python进阶之迭代器与迭代器切片教程
2020/01/29 Python
Sofft鞋官网:世界知名鞋类品牌
2017/03/28 全球购物
北美Newegg打造的全球尖货海购平台:tt海购
2018/09/28 全球购物
光声世纪笔试题目
2012/08/25 面试题
办公室保洁员岗位职责
2013/12/02 职场文书
高一地理教学反思
2014/01/18 职场文书
交通事故调解协议书
2014/04/16 职场文书
七夕情人节促销方案
2014/06/07 职场文书
市场营销专业自荐书
2014/06/10 职场文书
体育节口号
2014/06/19 职场文书
2014年师德师风自我剖析材料
2014/09/27 职场文书
教师政风行风评议心得体会
2014/10/21 职场文书
入伍通知书
2015/04/23 职场文书
小学见习报告
2015/06/23 职场文书
详解JS WebSocket断开原因和心跳机制
2021/05/07 Javascript
Java 多态分析
2022/04/26 Java/Android
MySQL示例讲解数据库约束以及表的设计
2022/06/16 MySQL