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


猜你喜欢
- 一、说明Boost.MPI 提供了 MPI 标准(消息传递接口)的接口。该标准简化了并发执行任务的程序的开发。您可以使用线程或通过共享内存或
- 做Java的面试题时遇到了以下这题,百度了一下Math.round()的修约规则,有的说是四舍五入,有的说是四舍六入,发现和我学分析化学时用
- 每日一笑下班和实习生一起回家,公交站等车,一乞丐把碗推向实习生乞讨。这时,实习生不慌不忙的说了句:“我不要你的钱,你这钱
- 四个主要操作类:JsonConverter 、JsonHelper 、JsonSplit 、AjaxResult一、JsonConverte
- 生成文字图片:/// <summary> /// 生成文字图片 /// &l
- 实践过程效果代码public partial class Form1 : Form{ public Form1()
- 修改Android FloatingActionButton的title的文字颜色及背景颜色实例详解首先看一张图片 我是在一个不错的开源的F
- 继承的概念继承是面向对象编程中的一个概念,它允许一个类(称为子类或派生类)继承另一个类(称为父类或基类)的属性和方法。子类在继承父类的同时也
- 前言在上一章节Spring和Mybatis整合的原理详解中有写到Spring和MyBatis整合时用到的Bean扫描是Spring本身提供的
- 实现GridView的横向滚动效果如下图:具体实现的代码•1. 主界面布局代码:activity_main.xml<?xml vers
- 本文演示以Spark作为分析引擎,Cassandra作为数据存储,而使用Spring Boot来开发驱动程序的示例。1.前置条件安装Spar
- 一、什么是Spring?Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器框架二、如何在程序中获取Spring配置的be
- 门面模式又叫外观模式(Facade Pattern),主要用于隐藏系统的复杂性,并向客户端提供了一个客户端可以访问系统的接口。我们知道电视剧
- 在安卓操作系统下对于 TextView 字体的支持非常有限,默认情况下 TextView 的 typeface 属性支持 "San
- 话不多说,跟着小编一起来看下吧using System;using System.Collections.Generic;using Sys
- 今天给大家讲讲android的目录实现方法,就像大家看到的小说目录一样,android 提供了ExpandableListView控件可以实
- 概念:LruCache什么是LruCache?LruCache实现原理是什么?这两个问题其实可以作为一个问题来回答,知道了什么是 LruCa
- RestTemplate设计是为了Spring更好的请求并解析Restful风格的接口返回值而设计的,通过这个类可以在请求接口时直接解析对应
- clone()和Cloneable接口clone顾名思义就是克隆,即,复制一个相等的对象,但是不同的引用地址。我们知道拿到一个对象的地址,只
- C#中List可谓是使用最广泛的一种数据类型了,使用他来规范数据时,往往会涉及到对数据的处理操作,相关处理数据方法也非常丰富,本文将简单介绍