【JVM系列】Class文件分析

💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。
img

  • 推荐:kwan 的首页,持续学习,不断总结,共同进步,活到老学到老
  • 导航
    • 檀越剑指大厂系列:全面总结 java 核心技术点,如集合,jvm,并发编程 redis,kafka,Spring,微服务,Netty 等
    • 常用开发工具系列:罗列常用的开发工具,如 IDEA,Mac,Alfred,electerm,Git,typora,apifox 等
    • 数据库系列:详细总结了常用数据库 mysql 技术点,以及工作中遇到的 mysql 问题等
    • 懒人运维系列:总结好用的命令,解放双手不香吗?能用一个命令完成绝不用两个操作
    • 数据结构与算法系列:总结数据结构和算法,不同类型针对性训练,提升编程思维,剑指大厂

非常期待和您一起在这个小小的网络世界里共同探索、学习和成长。💝💝💝 ✨✨ 欢迎订阅本专栏 ✨✨

博客目录

    • 一. class 文件介绍
      • 1.什么是 class 文件
      • 2.类文件的结构
      • 3.如何查看 class 文件
      • 4.文章背景
    • 二.数据准备
      • 1.实体类
      • 2.Class 文件
      • 3.查看 class 文件
    • 三.详细分析
      • 1.基本信息
      • 2.头部信息
      • 3.常量池总结
      • 4.常量池中的信息
      • 5.canEqual
      • 6.构造方法说明
      • 7.get 方法说明
      • 8.set 方法说明
      • 9.常量方法
      • 10.操作数栈和局部变量表区别?
      • 11.方法表总结
      • 12.方法的字节码指令
    • 四.Class 文件结构
      • 1.谈谈你对 class 文件的了解
      • 2.fields 和 attributes 区别
      • 3.DemoTest 解析
      • 4.class 文件中的魔数和主次版本号?
      • 5.为什么常量池计数器从 1 开始
      • 6.class 文件常量池中存放的是什么内容?
      • 7.常量池的项目类型?
      • 8.java 字段名和方法名长度限制?
      • 9.class 文件的访问标志作用?
      • 10.类索引、父类索引、接口索引集合作用
      • 11.class 文件的字段表存储的什么信息?
      • 12.class 文件的方法表存储的什么?
      • 13.class 文件中的属性表?

一. class 文件介绍

1.什么是 class 文件

Java 源代码经过编译器编译后产生的二进制文件称为 Java 类文件。这些类文件包含了与 Java 源代码对应的字节码,而字节码是一种中间代码,介于源代码和机器码之间。Java 类文件通常具有.class的扩展名。

当你编写 Java 程序时,你首先使用文本编辑器(如 Java 的开发工具,如 Eclipse、IntelliJ IDEA 等)编写源代码文件(扩展名为.java)。然后,你使用 Java 编译器(javac命令)将源代码编译成字节码,生成对应的 Java 类文件。最后,Java 虚拟机(JVM)会执行这些类文件中的字节码。

Java 类文件包含了类的结构信息、字段、方法以及其他一些与类相关的信息。这种中间表示形式使得 Java 具有平台无关性,因为一旦生成了字节码,它就可以在任何支持 Java 虚拟机的平台上运行,而无需重新编译。

2.类文件的结构

类文件的结构是由 Java 虚拟机规范定义的,包括魔数、版本信息、常量池、访问标志、类信息、字段信息、方法信息等部分。虽然类文件可以通过反编译工具(如javap、JD-GUI 等)进行查看和分析,但是这些工具仅提供了类的结构信息,而不包括原始的源代码。

总的来说,Java 类文件是 Java 程序的编译结果,其中包含了以字节码形式表示的程序的中间表示。这种设计使得 Java 能够实现跨平台的特性,因为字节码可以在任何支持 Java 虚拟机的计算机上运行。

3.如何查看 class 文件

要查看 Java 类文件的内容,可以使用多种工具。其中最常用的工具之一是javap命令,它是 Java Development Kit(JDK)中的一部分。以下是一些使用javap的基本示例:

  1. 使用 javap 查看类的成员变量和方法:

    javap YourClassName
    

    例如,如果你有一个名为MyClass的类文件,可以运行以下命令:

    javap MyClass
    

    这将显示类的成员变量和方法的信息。

  2. 使用 javap 查看详细信息:

    javap -verbose YourClassName
    

    这将显示更详细的类信息,包括常量池、字段描述符、方法描述符等。

  3. 将输出保存到文件:

    javap -verbose YourClassName > output.txt
    

    这将把javap的输出保存到一个名为output.txt的文件中,以便在文本编辑器中查看。

确保在运行这些命令之前,你已经安装了 Java Development Kit(JDK)并设置了相应的环境变量。如果你在命令行中运行这些命令时遇到问题,可能需要检查你的 Java 安装是否正确。

请注意,通过反编译工具查看类文件可能违反软件使用条款或法律规定,因此请确保你有权利查看和使用这些文件。

4.文章背景

之前也看过深入理解java虚拟机这本书,里面详细介绍了 class 文件的结构,自己也能说出来一些,但是仅限于八股文的地步,并没有了解书中每个 class 文件结构在实际代码中的含义,不能做到举一反三,完全停留在背八股的阶段。

通过本文详细的对 class 文件分析,结合深入理解java虚拟机这本书,可以更深刻的理解 class 文件结构,而不是停留在死记硬背的阶段。

二.数据准备

1.实体类

@Data
@ApiModel("chatgpt问答实体类")
public class Student extends Model<Student> implements Serializable {private static final long serialVersionUID = 1L;private static final String name = "qinyingjie1";private static String notFinal = "qinyingjie2";private final String nameNoStatic = "qinyingjie_nameNoStatic";private  String noneName = "qinyingjie_noneName";private static final Integer num = 20;private static final int intnum = 30;private static final String newStr = new String("qinyuchen");@ApiModelProperty("主键id")private Integer id;@ApiModelProperty("问题")private String question;@ApiModelProperty("回答")private String response;@ApiModelProperty("创建时间")private Date createTime;@ApiModelProperty("删除标识")private Integer isDelete;public static String getStr() {return "qyj";}
}

2.Class 文件

@ApiModel("chatgpt问答实体类")
public class Student extends Model<Student> implements Serializable {private static final long serialVersionUID = 1L;private static final String name = "qinyingjie1";private static String notFinal = "qinyingjie2";private final String nameNoStatic = "qinyingjie_nameNoStatic";private String noneName = "qinyingjie_noneName";private static final Integer num = 20;private static final int intnum = 30;private static final String newStr = new String("qinyuchen");@ApiModelProperty("主键id")private Integer id;@ApiModelProperty("问题")private String question;@ApiModelProperty("回答")private String response;@ApiModelProperty("创建时间")private Date createTime;@ApiModelProperty("删除标识")private Integer isDelete;public static String getStr() {return "qyj";}public Student() {}public String getNameNoStatic() {this.getClass();return "qinyingjie_nameNoStatic";}public String getNoneName() {return this.noneName;}public Integer getId() {return this.id;}public String getQuestion() {return this.question;}public String getResponse() {return this.response;}public Date getCreateTime() {return this.createTime;}public Integer getIsDelete() {return this.isDelete;}public void setNoneName(final String noneName) {this.noneName = noneName;}public void setId(final Integer id) {this.id = id;}public void setQuestion(final String question) {this.question = question;}public void setResponse(final String response) {this.response = response;}public void setCreateTime(final Date createTime) {this.createTime = createTime;}public void setIsDelete(final Integer isDelete) {this.isDelete = isDelete;}public boolean equals(final Object o) {if (o == this) {return true;} else if (!(o instanceof Student)) {return false;} else {Student other = (Student)o;if (!other.canEqual(this)) {return false;} else {label95: {Object this$id = this.getId();Object other$id = other.getId();if (this$id == null) {if (other$id == null) {break label95;}} else if (this$id.equals(other$id)) {break label95;}return false;}Object this$isDelete = this.getIsDelete();Object other$isDelete = other.getIsDelete();if (this$isDelete == null) {if (other$isDelete != null) {return false;}} else if (!this$isDelete.equals(other$isDelete)) {return false;}Object this$nameNoStatic = this.getNameNoStatic();Object other$nameNoStatic = other.getNameNoStatic();if (this$nameNoStatic == null) {if (other$nameNoStatic != null) {return false;}} else if (!this$nameNoStatic.equals(other$nameNoStatic)) {return false;}label74: {Object this$noneName = this.getNoneName();Object other$noneName = other.getNoneName();if (this$noneName == null) {if (other$noneName == null) {break label74;}} else if (this$noneName.equals(other$noneName)) {break label74;}return false;}label67: {Object this$question = this.getQuestion();Object other$question = other.getQuestion();if (this$question == null) {if (other$question == null) {break label67;}} else if (this$question.equals(other$question)) {break label67;}return false;}Object this$response = this.getResponse();Object other$response = other.getResponse();if (this$response == null) {if (other$response != null) {return false;}} else if (!this$response.equals(other$response)) {return false;}Object this$createTime = this.getCreateTime();Object other$createTime = other.getCreateTime();if (this$createTime == null) {if (other$createTime != null) {return false;}} else if (!this$createTime.equals(other$createTime)) {return false;}return true;}}}protected boolean canEqual(final Object other) {return other instanceof Student;}public int hashCode() {int PRIME = true;int result = 1;Object $id = this.getId();int result = result * 59 + ($id == null ? 43 : $id.hashCode());Object $isDelete = this.getIsDelete();result = result * 59 + ($isDelete == null ? 43 : $isDelete.hashCode());Object $nameNoStatic = this.getNameNoStatic();result = result * 59 + ($nameNoStatic == null ? 43 : $nameNoStatic.hashCode());Object $noneName = this.getNoneName();result = result * 59 + ($noneName == null ? 43 : $noneName.hashCode());Object $question = this.getQuestion();result = result * 59 + ($question == null ? 43 : $question.hashCode());Object $response = this.getResponse();result = result * 59 + ($response == null ? 43 : $response.hashCode());Object $createTime = this.getCreateTime();result = result * 59 + ($createTime == null ? 43 : $createTime.hashCode());return result;}public String toString() {return "Student(nameNoStatic=" + this.getNameNoStatic() + ", noneName=" + this.getNoneName() + ", id=" + this.getId() + ", question=" + this.getQuestion() + ", response=" + this.getResponse() + ", createTime=" + this.getCreateTime() + ", isDelete=" + this.getIsDelete() + ")";}
}

3.查看 class 文件

#查看class文件的详细信息
javap -verbose Student

Student的Class文件详细信息如下所示,下面进行逐行分析

警告: 二进制文件Student包含com.kwan.springbootkwan.entity.Student
Classfile /Users/qinyingjie/Documents/idea-workspace/vue-springboot-kwan/target/classes/com/kwan/springbootkwan/entity/Student.classLast modified 2023-12-2; size 5396 bytesMD5 checksum 9424b6c0d7363472a517799b5f5191b5Compiled from "Student.java"
public class com.kwan.springbootkwan.entity.Student extends com.baomidou.mybatisplus.extension.activerecord.Model<com.kwan.springbootkwan.entity.Student> implements java.io.Serializableminor version: 0major version: 52flags: ACC_PUBLICACC_SUPER
Constant pool:#1 = String             #148          // qyj  静态方法中的字符串#2 = Methodref          #45.#149      // com/baomidou/mybatisplus/extension/activerecord/Model."<init>":()V#3 = String             #150          // qinyingjie_nameNoStatic#4 = Fieldref           #13.#151      // com/kwan/springbootkwan/entity/Student.nameNoStatic:Ljava/lang/String;#5 = String             #152          // qinyingjie_noneName#6 = Fieldref           #13.#153      // com/kwan/springbootkwan/entity/Student.noneName:Ljava/lang/String;#7 = Methodref          #154.#155     // java/lang/Object.getClass:()Ljava/lang/Class;#8 = Fieldref           #13.#156      // com/kwan/springbootkwan/entity/Student.id:Ljava/lang/Integer;#9 = Fieldref           #13.#157      // com/kwan/springbootkwan/entity/Student.question:Ljava/lang/String;#10 = Fieldref           #13.#158      // com/kwan/springbootkwan/entity/Student.response:Ljava/lang/String;#11 = Fieldref           #13.#159      // com/kwan/springbootkwan/entity/Student.createTime:Ljava/util/Date;#12 = Fieldref           #13.#160      // com/kwan/springbootkwan/entity/Student.isDelete:Ljava/lang/Integer;#13 = Class              #161          // com/kwan/springbootkwan/entity/Student#14 = Methodref          #13.#162      // com/kwan/springbootkwan/entity/Student.canEqual:(Ljava/lang/Object;)Z#15 = Methodref          #13.#163      // com/kwan/springbootkwan/entity/Student.getId:()Ljava/lang/Integer;#16 = Methodref          #154.#164     // java/lang/Object.equals:(Ljava/lang/Object;)Z#17 = Methodref          #13.#165      // com/kwan/springbootkwan/entity/Student.getIsDelete:()Ljava/lang/Integer;#18 = Methodref          #13.#166      // com/kwan/springbootkwan/entity/Student.getNameNoStatic:()Ljava/lang/String;#19 = Methodref          #13.#167      // com/kwan/springbootkwan/entity/Student.getNoneName:()Ljava/lang/String;#20 = Methodref          #13.#168      // com/kwan/springbootkwan/entity/Student.getQuestion:()Ljava/lang/String;#21 = Methodref          #13.#169      // com/kwan/springbootkwan/entity/Student.getResponse:()Ljava/lang/String;#22 = Methodref          #13.#170      // com/kwan/springbootkwan/entity/Student.getCreateTime:()Ljava/util/Date;#23 = Methodref          #154.#171     // java/lang/Object.hashCode:()I#24 = Class              #172          // java/lang/StringBuilder#25 = Methodref          #24.#149      // java/lang/StringBuilder."<init>":()V#26 = String             #173          // Student(nameNoStatic=#27 = Methodref          #24.#174      // java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;#28 = String             #175          // , noneName=#29 = String             #176          // , id=#30 = Methodref          #24.#177      // java/lang/StringBuilder.append:(Ljava/lang/Object;)Ljava/lang/StringBuilder;#31 = String             #178          // , question=#32 = String             #179          // , response=#33 = String             #180          // , createTime=#34 = String             #181          // , isDelete=#35 = String             #182          // )#36 = Methodref          #24.#183      // java/lang/StringBuilder.toString:()Ljava/lang/String;#37 = String             #184          // qinyingjie2#38 = Fieldref           #13.#185      // com/kwan/springbootkwan/entity/Student.notFinal:Ljava/lang/String;#39 = Methodref          #186.#187     // java/lang/Integer.valueOf:(I)Ljava/lang/Integer;#40 = Fieldref           #13.#188      // com/kwan/springbootkwan/entity/Student.num:Ljava/lang/Integer;#41 = Class              #189          // java/lang/String#42 = String             #190          // qinyuchen#43 = Methodref          #41.#191      // java/lang/String."<init>":(Ljava/lang/String;)V#44 = Fieldref           #13.#192      // com/kwan/springbootkwan/entity/Student.newStr:Ljava/lang/String;#45 = Class              #193          // com/baomidou/mybatisplus/extension/activerecord/Model#46 = Class              #194          // java/io/Serializable#47 = Utf8               serialVersionUID#48 = Utf8               J#49 = Utf8               ConstantValue#50 = Long               1l#52 = Utf8               name#53 = Utf8               Ljava/lang/String;#54 = String             #195          // qinyingjie1#55 = Utf8               notFinal#56 = Utf8               nameNoStatic#57 = Utf8               noneName#58 = Utf8               num#59 = Utf8               Ljava/lang/Integer;#60 = Utf8               intnum#61 = Utf8               I#62 = Integer            30#63 = Utf8               newStr#64 = Utf8               id#65 = Utf8               RuntimeVisibleAnnotations#66 = Utf8               Lio/swagger/annotations/ApiModelProperty;#67 = Utf8               value#68 = Utf8               主键id#69 = Utf8               question#70 = Utf8               问题#71 = Utf8               response#72 = Utf8               回答#73 = Utf8               createTime#74 = Utf8               Ljava/util/Date;#75 = Utf8               创建时间#76 = Utf8               isDelete#77 = Utf8               删除标识#78 = Utf8               getStr#79 = Utf8               ()Ljava/lang/String;#80 = Utf8               Code#81 = Utf8               LineNumberTable#82 = Utf8               <init>#83 = Utf8               ()V#84 = Utf8               LocalVariableTable#85 = Utf8               this#86 = Utf8               Lcom/kwan/springbootkwan/entity/Student;#87 = Utf8               getNameNoStatic#88 = Utf8               getNoneName#89 = Utf8               getId#90 = Utf8               ()Ljava/lang/Integer;#91 = Utf8               getQuestion#92 = Utf8               getResponse#93 = Utf8               getCreateTime#94 = Utf8               ()Ljava/util/Date;#95 = Utf8               getIsDelete#96 = Utf8               setNoneName#97 = Utf8               (Ljava/lang/String;)V#98 = Utf8               MethodParameters#99 = Utf8               setId#100 = Utf8               (Ljava/lang/Integer;)V#101 = Utf8               setQuestion#102 = Utf8               setResponse#103 = Utf8               setCreateTime#104 = Utf8               (Ljava/util/Date;)V#105 = Utf8               setIsDelete#106 = Utf8               equals#107 = Utf8               (Ljava/lang/Object;)Z#108 = Utf8               o#109 = Utf8               Ljava/lang/Object;#110 = Utf8               other#111 = Utf8               this$id#112 = Utf8               other$id#113 = Utf8               this$isDelete#114 = Utf8               other$isDelete#115 = Utf8               this$nameNoStatic#116 = Utf8               other$nameNoStatic#117 = Utf8               this$noneName#118 = Utf8               other$noneName#119 = Utf8               this$question#120 = Utf8               other$question#121 = Utf8               this$response#122 = Utf8               other$response#123 = Utf8               this$createTime#124 = Utf8               other$createTime#125 = Utf8               StackMapTable#126 = Class              #161          // com/kwan/springbootkwan/entity/Student#127 = Class              #196          // java/lang/Object#128 = Utf8               canEqual#129 = Utf8               hashCode#130 = Utf8               ()I#131 = Utf8               PRIME#132 = Utf8               result#133 = Utf8               $id#134 = Utf8               $isDelete#135 = Utf8               $nameNoStatic#136 = Utf8               $noneName#137 = Utf8               $question#138 = Utf8               $response#139 = Utf8               $createTime#140 = Utf8               toString#141 = Utf8               <clinit>#142 = Utf8               Signature#143 = Utf8               Lcom/baomidou/mybatisplus/extension/activerecord/Model<Lcom/kwan/springbootkwan/entity/Student;>;Ljava/io/Serializable;#144 = Utf8               SourceFile#145 = Utf8               Student.java#146 = Utf8               Lio/swagger/annotations/ApiModel;#147 = Utf8               chatgpt问答实体类#148 = Utf8               qyj#149 = NameAndType        #82:#83       // "<init>":()V#150 = Utf8               qinyingjie_nameNoStatic#151 = NameAndType        #56:#53       // nameNoStatic:Ljava/lang/String;#152 = Utf8               qinyingjie_noneName#153 = NameAndType        #57:#53       // noneName:Ljava/lang/String;#154 = Class              #196          // java/lang/Object#155 = NameAndType        #197:#198     // getClass:()Ljava/lang/Class;#156 = NameAndType        #64:#59       // id:Ljava/lang/Integer;#157 = NameAndType        #69:#53       // question:Ljava/lang/String;#158 = NameAndType        #71:#53       // response:Ljava/lang/String;#159 = NameAndType        #73:#74       // createTime:Ljava/util/Date;#160 = NameAndType        #76:#59       // isDelete:Ljava/lang/Integer;#161 = Utf8               com/kwan/springbootkwan/entity/Student#162 = NameAndType        #128:#107     // canEqual:(Ljava/lang/Object;)Z#163 = NameAndType        #89:#90       // getId:()Ljava/lang/Integer;#164 = NameAndType        #106:#107     // equals:(Ljava/lang/Object;)Z#165 = NameAndType        #95:#90       // getIsDelete:()Ljava/lang/Integer;#166 = NameAndType        #87:#79       // getNameNoStatic:()Ljava/lang/String;#167 = NameAndType        #88:#79       // getNoneName:()Ljava/lang/String;#168 = NameAndType        #91:#79       // getQuestion:()Ljava/lang/String;#169 = NameAndType        #92:#79       // getResponse:()Ljava/lang/String;#170 = NameAndType        #93:#94       // getCreateTime:()Ljava/util/Date;#171 = NameAndType        #129:#130     // hashCode:()I#172 = Utf8               java/lang/StringBuilder#173 = Utf8               Student(nameNoStatic=#174 = NameAndType        #199:#200     // append:(Ljava/lang/String;)Ljava/lang/StringBuilder;#175 = Utf8               , noneName=#176 = Utf8               , id=#177 = NameAndType        #199:#201     // append:(Ljava/lang/Object;)Ljava/lang/StringBuilder;#178 = Utf8               , question=#179 = Utf8               , response=#180 = Utf8               , createTime=#181 = Utf8               , isDelete=#182 = Utf8               )#183 = NameAndType        #140:#79      // toString:()Ljava/lang/String;#184 = Utf8               qinyingjie2#185 = NameAndType        #55:#53       // notFinal:Ljava/lang/String;#186 = Class              #202          // java/lang/Integer#187 = NameAndType        #203:#204     // valueOf:(I)Ljava/lang/Integer;#188 = NameAndType        #58:#59       // num:Ljava/lang/Integer;#189 = Utf8               java/lang/String#190 = Utf8               qinyuchen#191 = NameAndType        #82:#97       // "<init>":(Ljava/lang/String;)V#192 = NameAndType        #63:#53       // newStr:Ljava/lang/String;#193 = Utf8               com/baomidou/mybatisplus/extension/activerecord/Model#194 = Utf8               java/io/Serializable#195 = Utf8               qinyingjie1#196 = Utf8               java/lang/Object#197 = Utf8               getClass#198 = Utf8               ()Ljava/lang/Class;#199 = Utf8               append#200 = Utf8               (Ljava/lang/String;)Ljava/lang/StringBuilder;#201 = Utf8               (Ljava/lang/Object;)Ljava/lang/StringBuilder;#202 = Utf8               java/lang/Integer#203 = Utf8               valueOf#204 = Utf8               (I)Ljava/lang/Integer;
{public static java.lang.String getStr();descriptor: ()Ljava/lang/String;flags: ACC_PUBLICACC_STATICCode:stack=1, locals=0, args_size=00: ldc           #1                  // String qyj2: areturnLineNumberTable:line 43: 0public com.kwan.springbootkwan.entity.Student();descriptor: ()Vflags: ACC_PUBLICCode:stack=2, locals=1, args_size=10: aload_01: invokespecial #2                  // Method com/baomidou/mybatisplus/extension/activerecord/Model."<init>":()V4: aload_05: ldc           #3                  // String qinyingjie_nameNoStatic7: putfield      #4                  // Field nameNoStatic:Ljava/lang/String;10: aload_011: ldc           #5                  // String qinyingjie_noneName13: putfield      #6                  // Field noneName:Ljava/lang/String;16: returnLineNumberTable:line 12: 0line 21: 4line 23: 10LocalVariableTable:Start  Length  Slot  Name   Signature0      17     0  this   Lcom/kwan/springbootkwan/entity/Student;public java.lang.String getNameNoStatic();descriptor: ()Ljava/lang/String;flags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: invokevirtual #7                  // Method java/lang/Object.getClass:()Ljava/lang/Class;4: pop5: ldc           #3                  // String qinyingjie_nameNoStatic7: areturnLineNumberTable:line 21: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       8     0  this   Lcom/kwan/springbootkwan/entity/Student;public java.lang.String getNoneName();descriptor: ()Ljava/lang/String;flags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: getfield      #6                  // Field noneName:Ljava/lang/String;4: areturnLineNumberTable:line 23: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;public java.lang.Integer getId();descriptor: ()Ljava/lang/Integer;flags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: getfield      #8                  // Field id:Ljava/lang/Integer;4: areturnLineNumberTable:line 32: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;public java.lang.String getQuestion();descriptor: ()Ljava/lang/String;flags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: getfield      #9                  // Field question:Ljava/lang/String;4: areturnLineNumberTable:line 34: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;public java.lang.String getResponse();descriptor: ()Ljava/lang/String;flags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: getfield      #10                 // Field response:Ljava/lang/String;4: areturnLineNumberTable:line 36: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;public java.util.Date getCreateTime();descriptor: ()Ljava/util/Date;flags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: getfield      #11                 // Field createTime:Ljava/util/Date;4: areturnLineNumberTable:line 38: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;public java.lang.Integer getIsDelete();descriptor: ()Ljava/lang/Integer;flags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: getfield      #12                 // Field isDelete:Ljava/lang/Integer;4: areturnLineNumberTable:line 40: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;public void setNoneName(java.lang.String);descriptor: (Ljava/lang/String;)Vflags: ACC_PUBLICCode:stack=2, locals=2, args_size=20: aload_01: aload_12: putfield      #6                  // Field noneName:Ljava/lang/String;5: returnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       6     0  this   Lcom/kwan/springbootkwan/entity/Student;0       6     1 noneName   Ljava/lang/String;MethodParameters:Name                           FlagsnoneName                       finalpublic void setId(java.lang.Integer);descriptor: (Ljava/lang/Integer;)Vflags: ACC_PUBLICCode:stack=2, locals=2, args_size=20: aload_01: aload_12: putfield      #8                  // Field id:Ljava/lang/Integer;5: returnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       6     0  this   Lcom/kwan/springbootkwan/entity/Student;0       6     1    id   Ljava/lang/Integer;MethodParameters:Name                           Flagsid                             finalpublic void setQuestion(java.lang.String);descriptor: (Ljava/lang/String;)Vflags: ACC_PUBLICCode:stack=2, locals=2, args_size=20: aload_01: aload_12: putfield      #9                  // Field question:Ljava/lang/String;5: returnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       6     0  this   Lcom/kwan/springbootkwan/entity/Student;0       6     1 question   Ljava/lang/String;MethodParameters:Name                           Flagsquestion                       finalpublic void setResponse(java.lang.String);descriptor: (Ljava/lang/String;)Vflags: ACC_PUBLICCode:stack=2, locals=2, args_size=20: aload_01: aload_12: putfield      #10                 // Field response:Ljava/lang/String;5: returnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       6     0  this   Lcom/kwan/springbootkwan/entity/Student;0       6     1 response   Ljava/lang/String;MethodParameters:Name                           Flagsresponse                       finalpublic void setCreateTime(java.util.Date);descriptor: (Ljava/util/Date;)Vflags: ACC_PUBLICCode:stack=2, locals=2, args_size=20: aload_01: aload_12: putfield      #11                 // Field createTime:Ljava/util/Date;5: returnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       6     0  this   Lcom/kwan/springbootkwan/entity/Student;0       6     1 createTime   Ljava/util/Date;MethodParameters:Name                           FlagscreateTime                     finalpublic void setIsDelete(java.lang.Integer);descriptor: (Ljava/lang/Integer;)Vflags: ACC_PUBLICCode:stack=2, locals=2, args_size=20: aload_01: aload_12: putfield      #12                 // Field isDelete:Ljava/lang/Integer;5: returnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       6     0  this   Lcom/kwan/springbootkwan/entity/Student;0       6     1 isDelete   Ljava/lang/Integer;MethodParameters:Name                           FlagsisDelete                       finalpublic boolean equals(java.lang.Object);descriptor: (Ljava/lang/Object;)Zflags: ACC_PUBLICCode:stack=2, locals=17, args_size=20: aload_11: aload_02: if_acmpne     75: iconst_16: ireturn7: aload_18: instanceof    #13                 // class com/kwan/springbootkwan/entity/Student11: ifne          1614: iconst_015: ireturn16: aload_117: checkcast     #13                 // class com/kwan/springbootkwan/entity/Student20: astore_221: aload_222: aload_023: invokevirtual #14                 // Method canEqual:(Ljava/lang/Object;)Z26: ifne          3129: iconst_030: ireturn31: aload_032: invokevirtual #15                 // Method getId:()Ljava/lang/Integer;35: astore_336: aload_237: invokevirtual #15                 // Method getId:()Ljava/lang/Integer;40: astore        442: aload_343: ifnonnull     5446: aload         448: ifnull        6551: goto          6354: aload_355: aload         457: invokevirtual #16                 // Method java/lang/Object.equals:(Ljava/lang/Object;)Z60: ifne          6563: iconst_064: ireturn65: aload_066: invokevirtual #17                 // Method getIsDelete:()Ljava/lang/Integer;69: astore        571: aload_272: invokevirtual #17                 // Method getIsDelete:()Ljava/lang/Integer;75: astore        677: aload         579: ifnonnull     9082: aload         684: ifnull        10287: goto          10090: aload         592: aload         694: invokevirtual #16                 // Method java/lang/Object.equals:(Ljava/lang/Object;)Z97: ifne          102100: iconst_0101: ireturn102: aload_0103: invokevirtual #18                 // Method getNameNoStatic:()Ljava/lang/String;106: astore        7108: aload_2109: invokevirtual #18                 // Method getNameNoStatic:()Ljava/lang/String;112: astore        8114: aload         7116: ifnonnull     127119: aload         8121: ifnull        139124: goto          137127: aload         7129: aload         8131: invokevirtual #16                 // Method java/lang/Object.equals:(Ljava/lang/Object;)Z134: ifne          139137: iconst_0138: ireturn139: aload_0140: invokevirtual #19                 // Method getNoneName:()Ljava/lang/String;143: astore        9145: aload_2146: invokevirtual #19                 // Method getNoneName:()Ljava/lang/String;149: astore        10151: aload         9153: ifnonnull     164156: aload         10158: ifnull        176161: goto          174164: aload         9166: aload         10168: invokevirtual #16                 // Method java/lang/Object.equals:(Ljava/lang/Object;)Z171: ifne          176174: iconst_0175: ireturn176: aload_0177: invokevirtual #20                 // Method getQuestion:()Ljava/lang/String;180: astore        11182: aload_2183: invokevirtual #20                 // Method getQuestion:()Ljava/lang/String;186: astore        12188: aload         11190: ifnonnull     201193: aload         12195: ifnull        213198: goto          211201: aload         11203: aload         12205: invokevirtual #16                 // Method java/lang/Object.equals:(Ljava/lang/Object;)Z208: ifne          213211: iconst_0212: ireturn213: aload_0214: invokevirtual #21                 // Method getResponse:()Ljava/lang/String;217: astore        13219: aload_2220: invokevirtual #21                 // Method getResponse:()Ljava/lang/String;223: astore        14225: aload         13227: ifnonnull     238230: aload         14232: ifnull        250235: goto          248238: aload         13240: aload         14242: invokevirtual #16                 // Method java/lang/Object.equals:(Ljava/lang/Object;)Z245: ifne          250248: iconst_0249: ireturn250: aload_0251: invokevirtual #22                 // Method getCreateTime:()Ljava/util/Date;254: astore        15256: aload_2257: invokevirtual #22                 // Method getCreateTime:()Ljava/util/Date;260: astore        16262: aload         15264: ifnonnull     275267: aload         16269: ifnull        287272: goto          285275: aload         15277: aload         16279: invokevirtual #16                 // Method java/lang/Object.equals:(Ljava/lang/Object;)Z282: ifne          287285: iconst_0286: ireturn287: iconst_1288: ireturnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0     289     0  this   Lcom/kwan/springbootkwan/entity/Student;0     289     1     o   Ljava/lang/Object;21     268     2 other   Lcom/kwan/springbootkwan/entity/Student;36     253     3 this$id   Ljava/lang/Object;42     247     4 other$id   Ljava/lang/Object;71     218     5 this$isDelete   Ljava/lang/Object;77     212     6 other$isDelete   Ljava/lang/Object;108     181     7 this$nameNoStatic   Ljava/lang/Object;114     175     8 other$nameNoStatic   Ljava/lang/Object;145     144     9 this$noneName   Ljava/lang/Object;151     138    10 other$noneName   Ljava/lang/Object;182     107    11 this$question   Ljava/lang/Object;188     101    12 other$question   Ljava/lang/Object;219      70    13 this$response   Ljava/lang/Object;225      64    14 other$response   Ljava/lang/Object;256      33    15 this$createTime   Ljava/lang/Object;262      27    16 other$createTime   Ljava/lang/Object;StackMapTable: number_of_entries = 24frame_type = 7 /* same */frame_type = 8 /* same */frame_type = 252 /* append */offset_delta = 14locals = [ class com/kwan/springbootkwan/entity/Student ]frame_type = 253 /* append */offset_delta = 22locals = [ class java/lang/Objectclass java/lang/Object ]frame_type = 8 /* same */frame_type = 1 /* same */frame_type = 253 /* append */offset_delta = 24locals = [ class java/lang/Objectclass java/lang/Object ]frame_type = 9 /* same */frame_type = 1 /* same */frame_type = 253 /* append */offset_delta = 24locals = [ class java/lang/Objectclass java/lang/Object ]frame_type = 9 /* same */frame_type = 1 /* same */frame_type = 253 /* append */offset_delta = 24locals = [ class java/lang/Objectclass java/lang/Object ]frame_type = 9 /* same */frame_type = 1 /* same */frame_type = 253 /* append */offset_delta = 24locals = [ class java/lang/Objectclass java/lang/Object ]frame_type = 9 /* same */frame_type = 1 /* same */frame_type = 253 /* append */offset_delta = 24locals = [ class java/lang/Objectclass java/lang/Object ]frame_type = 9 /* same */frame_type = 1 /* same */frame_type = 253 /* append */offset_delta = 24locals = [ class java/lang/Objectclass java/lang/Object ]frame_type = 9 /* same */frame_type = 1 /* same */MethodParameters:Name                           Flagso                              finalprotected boolean canEqual(java.lang.Object);descriptor: (Ljava/lang/Object;)Zflags: ACC_PROTECTEDCode:stack=1, locals=2, args_size=20: aload_11: instanceof    #13                 // class com/kwan/springbootkwan/entity/Student4: ireturnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;0       5     1 other   Ljava/lang/Object;MethodParameters:Name                           Flagsother                          finalpublic int hashCode();descriptor: ()Iflags: ACC_PUBLICCode:stack=2, locals=10, args_size=10: bipush        592: istore_13: iconst_14: istore_25: aload_06: invokevirtual #15                 // Method getId:()Ljava/lang/Integer;9: astore_310: iload_211: bipush        5913: imul14: aload_315: ifnonnull     2318: bipush        4320: goto          2723: aload_324: invokevirtual #23                 // Method java/lang/Object.hashCode:()I27: iadd28: istore_229: aload_030: invokevirtual #17                 // Method getIsDelete:()Ljava/lang/Integer;33: astore        435: iload_236: bipush        5938: imul39: aload         441: ifnonnull     4944: bipush        4346: goto          5449: aload         451: invokevirtual #23                 // Method java/lang/Object.hashCode:()I54: iadd55: istore_256: aload_057: invokevirtual #18                 // Method getNameNoStatic:()Ljava/lang/String;60: astore        562: iload_263: bipush        5965: imul66: aload         568: ifnonnull     7671: bipush        4373: goto          8176: aload         578: invokevirtual #23                 // Method java/lang/Object.hashCode:()I81: iadd82: istore_283: aload_084: invokevirtual #19                 // Method getNoneName:()Ljava/lang/String;87: astore        689: iload_290: bipush        5992: imul93: aload         695: ifnonnull     10398: bipush        43100: goto          108103: aload         6105: invokevirtual #23                 // Method java/lang/Object.hashCode:()I108: iadd109: istore_2110: aload_0111: invokevirtual #20                 // Method getQuestion:()Ljava/lang/String;114: astore        7116: iload_2117: bipush        59119: imul120: aload         7122: ifnonnull     130125: bipush        43127: goto          135130: aload         7132: invokevirtual #23                 // Method java/lang/Object.hashCode:()I135: iadd136: istore_2137: aload_0138: invokevirtual #21                 // Method getResponse:()Ljava/lang/String;141: astore        8143: iload_2144: bipush        59146: imul147: aload         8149: ifnonnull     157152: bipush        43154: goto          162157: aload         8159: invokevirtual #23                 // Method java/lang/Object.hashCode:()I162: iadd163: istore_2164: aload_0165: invokevirtual #22                 // Method getCreateTime:()Ljava/util/Date;168: astore        9170: iload_2171: bipush        59173: imul174: aload         9176: ifnonnull     184179: bipush        43181: goto          189184: aload         9186: invokevirtual #23                 // Method java/lang/Object.hashCode:()I189: iadd190: istore_2191: iload_2192: ireturnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0     193     0  this   Lcom/kwan/springbootkwan/entity/Student;3     190     1 PRIME   I5     188     2 result   I10     183     3   $id   Ljava/lang/Object;35     158     4 $isDelete   Ljava/lang/Object;62     131     5 $nameNoStatic   Ljava/lang/Object;89     104     6 $noneName   Ljava/lang/Object;116      77     7 $question   Ljava/lang/Object;143      50     8 $response   Ljava/lang/Object;170      23     9 $createTime   Ljava/lang/Object;StackMapTable: number_of_entries = 14frame_type = 255 /* full_frame */offset_delta = 23locals = [ class com/kwan/springbootkwan/entity/Studentintintclass java/lang/Object ]stack = [ int ]frame_type = 255 /* full_frame */offset_delta = 3locals = [ class com/kwan/springbootkwan/entity/Studentintintclass java/lang/Object ]stack = [ intint ]frame_type = 255 /* full_frame */offset_delta = 21locals = [ class com/kwan/springbootkwan/entity/Studentintintclass java/lang/Objectclass java/lang/Object ]stack = [ int ]frame_type = 255 /* full_frame */offset_delta = 4locals = [ class com/kwan/springbootkwan/entity/Studentintintclass java/lang/Objectclass java/lang/Object ]stack = [ intint ]frame_type = 255 /* full_frame */offset_delta = 21locals = [ class com/kwan/springbootkwan/entity/Studentintintclass java/lang/Objectclass java/lang/Objectclass java/lang/Object ]stack = [ int ]frame_type = 255 /* full_frame */offset_delta = 4locals = [ class com/kwan/springbootkwan/entity/Studentintintclass java/lang/Objectclass java/lang/Objectclass java/lang/Object ]stack = [ intint ]frame_type = 255 /* full_frame */offset_delta = 21locals = [ class com/kwan/springbootkwan/entity/Studentintintclass java/lang/Objectclass java/lang/Objectclass java/lang/Objectclass java/lang/Object ]stack = [ int ]frame_type = 255 /* full_frame */offset_delta = 4locals = [ class com/kwan/springbootkwan/entity/Studentintintclass java/lang/Objectclass java/lang/Object, class java/lang/Object, class java/lang/Object ]stack = [ int, int ]frame_type = 255 /* full_frame */offset_delta = 21locals = [ class com/kwan/springbootkwan/entity/Student, int, int, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object ]stack = [ int ]frame_type = 255 /* full_frame */offset_delta = 4locals = [ class com/kwan/springbootkwan/entity/Student, int, int, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object ]stack = [ int, int ]frame_type = 255 /* full_frame */offset_delta = 21locals = [ class com/kwan/springbootkwan/entity/Student, int, int, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object ]stack = [ int ]frame_type = 255 /* full_frame */offset_delta = 4locals = [ class com/kwan/springbootkwan/entity/Student, int, int, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object ]stack = [ int, int ]frame_type = 255 /* full_frame */offset_delta = 21locals = [ class com/kwan/springbootkwan/entity/Student, int, int, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object ]stack = [ int ]frame_type = 255 /* full_frame */offset_delta = 4locals = [ class com/kwan/springbootkwan/entity/Student, int, int, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object, class java/lang/Object ]stack = [ int, int ]public java.lang.String toString();descriptor: ()Ljava/lang/String;flags: ACC_PUBLICCode:stack=2, locals=1, args_size=10: new           #24                 // class java/lang/StringBuilder3: dup4: invokespecial #25                 // Method java/lang/StringBuilder."<init>":()V7: ldc           #26                 // String Student(nameNoStatic=9: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;12: aload_013: invokevirtual #18                 // Method getNameNoStatic:()Ljava/lang/String;16: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;19: ldc           #28                 // String , noneName=21: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;24: aload_025: invokevirtual #19                 // Method getNoneName:()Ljava/lang/String;28: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;31: ldc           #29                 // String , id=33: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;36: aload_037: invokevirtual #15                 // Method getId:()Ljava/lang/Integer;40: invokevirtual #30                 // Method java/lang/StringBuilder.append:(Ljava/lang/Object;)Ljava/lang/StringBuilder;43: ldc           #31                 // String , question=45: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;48: aload_049: invokevirtual #20                 // Method getQuestion:()Ljava/lang/String;52: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;55: ldc           #32                 // String , response=57: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;60: aload_061: invokevirtual #21                 // Method getResponse:()Ljava/lang/String;64: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;67: ldc           #33                 // String , createTime=69: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;72: aload_073: invokevirtual #22                 // Method getCreateTime:()Ljava/util/Date;76: invokevirtual #30                 // Method java/lang/StringBuilder.append:(Ljava/lang/Object;)Ljava/lang/StringBuilder;79: ldc           #34                 // String , isDelete=81: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;84: aload_085: invokevirtual #17                 // Method getIsDelete:()Ljava/lang/Integer;88: invokevirtual #30                 // Method java/lang/StringBuilder.append:(Ljava/lang/Object;)Ljava/lang/StringBuilder;91: ldc           #35                 // String )93: invokevirtual #27                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;96: invokevirtual #36                 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;99: areturnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0     100     0  this   Lcom/kwan/springbootkwan/entity/Student;static {};descriptor: ()Vflags: ACC_STATICCode:stack=3, locals=0, args_size=00: ldc           #37                 // String qinyingjie22: putstatic     #38                 // Field notFinal:Ljava/lang/String;5: bipush        207: invokestatic  #39                 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;10: putstatic     #40                 // Field num:Ljava/lang/Integer;13: new           #41                 // class java/lang/String16: dup17: ldc           #42                 // String qinyuchen19: invokespecial #43                 // Method java/lang/String."<init>":(Ljava/lang/String;)V22: putstatic     #44                 // Field newStr:Ljava/lang/String;25: returnLineNumberTable:line 19: 0line 25: 5line 29: 13
}
Signature: #143                         // Lcom/baomidou/mybatisplus/extension/activerecord/Model<Lcom/kwan/springbootkwan/entity/Student;>;Ljava/io/Serializable;
SourceFile: "Student.java"
RuntimeVisibleAnnotations:

三.详细分析

1.基本信息

  1. 警告信息显示源 class 文件名称
  2. Classfile 显示了 class 文件的详细地址信息
  3. 最后一次的更新时间
  4. class 文件的字节大小
  5. MD5 值 checksum
  6. class 文件的编译来源文件
警告: 二进制文件Student包含com.kwan.springbootkwan.entity.Student
Classfile /Users/qinyingjie/Documents/idea-workspace/vue-springboot-kwan/target/classes/com/kwan/springbootkwan/entity/Student.classLast modified 2023-12-2; size 4370 bytesMD5 checksum 7683c822cf8c123129e8867436bb50efCompiled from "Student.java"

2.头部信息

  1. class 文件名
  2. 继承信息
  3. 实现接口信息
  4. 次版本号
  5. 主版本号
  6. 访问标志 flags
public class com.kwan.springbootkwan.entity.Student extends com.baomidou.mybatisplus.extension.activerecord.Model<com.kwan.springbootkwan.entity.Student> implements java.io.Serializableminor version: 0major version: 52flags: ACC_PUBLIC, ACC_SUPER

在 Java 类文件的访问标志(Access Flags)中,ACC_PUBLICACC_SUPER是两个常见的标志,它们用于描述类的访问修饰符和其他特性。

  1. ACC_PUBLIC:

    • 意义:表示该类是公共的,可以被其他类访问。
    • 解释:具有ACC_PUBLIC标志的类对于其他类是可见的,可以在同一包中或不同包中的其他类中访问。这是 Java 中最广泛使用的类访问修饰符。
  2. ACC_SUPER:

    • 意义:表示该类调用超类的方法时使用invokespecial指令。
    • 解释:ACC_SUPER标志在类文件中用于指示是否调用超类(父类)的方法时使用invokespecial指令。在 Java 虚拟机规范的早期版本中,用于实现虚拟方法调用的指令是invokespecial,它被用来调用超类中的构造方法。后来,随着 Java 语言的发展,invokespecial也用于调用私有方法、实例初始化方法和初始化方法。

    注意:

    • 在 Java 5 及以后的版本中,虚拟机对于ACC_SUPER标志的使用发生了变化,实际上这个标志的作用不再与invokespecial指令有关。它仍然会被设置,但是在实际的虚拟机实现中可能没有直接的影响。

总体而言,ACC_PUBLIC表示类是公共的,可以被其他类访问,而ACC_SUPER在过去用于指示虚拟机在方法调用时使用invokespecial指令。

3.常量池总结

常量池是类文件中存储常量的表,包括类的如下几种信息:

字段(Fieldref):

  • 类的字段对应的Fieldref(字段引用常量)也存放在常量池中。

方法名(类方法和父类方法Methodref):

  • 父类方法的 init 方法会被当做Methodref(方法引用常量)存放在常量池中。
  • 本类中方法引用,如 canEqual 方法的引用也会放在常量池中,主要是重写 equals 方法。

方法描述符(Utf8):

  • setId 方法的参数是 Integer,返回值是 void,会作为一个整体以 UTF-8 编码的方式存在常量池。

字符串(String或者Utf8):

  • new 出来的 qinyuchen 这个字符串会以字符串的形式存放在常量池。
  • ApiModelProperty 注解的描述会以字符串的形式存储在常量池中,区别是以 UTF-8 编码的。
  • toString 方法模版生成的字符串也是以 utf-8 编码的形式存在常量池中。
  • toString 底层是使用 StringBuilder,然后用到了 append 方法,append 方法名也会存在常量池。
  • 方法表中的属性表中的 ConstantValue 属性存储了 static final 修饰的 utf-8 编码的常量,可以直接被类访问,如qinyingjie1
  • 只被 static 修饰的字符串以 String 形式存储在常量池中,如qinyingjie2
  • 只被 final 修饰的字符串以 String 形式存储在常量池中,如qinyingjie_nameNoStatic
  • 没有被 static final 修饰的字符串以 String 形式存储在常量池中,如qinyingjie_noneName

类引用(Class):

  • Model 类是继承的类,类引用常量也会放在常量池中。
  • 使用到了 Integer,Integer 的类引用也会存在常量池。

4.常量池中的信息

  • 常量池中的第一位,如果有静态方法,并且静态方法中有字符串,则第一位会给这个字符串。
  • 如果没有静态方法,则第一位给父类的构造方法。

以下是一些常量池中的信息(这一部分的class常量池文件是没加静态方法getStr的文件):

  1. #1 = Methodref #35.#126:

    • 这是一个方法引用(Methodref)常量,表示对com/baomidou/mybatisplus/extension/activerecord/Model类中名为<init>且没有参数的构造方法的引用。
  2. #2 = Fieldref #7.#127:

    • 这是一个字段引用(Fieldref)常量,表示对com/kwan/springbootkwan/entity/Student类中名为id的字段的引用。
  3. #8 = Methodref #7.#133:

    • 这是一个方法引用(Methodref)常量,表示对com/kwan/springbootkwan/entity/Student类中名为canEqual的方法的引用。
  4. #32 = String #158:

    • 这是一个字符串常量,表示字符串"qinyuchen"。
  5. #35 = Class #161:

    • 这是一个类引用(Class)常量,表示对com/baomidou/mybatisplus/extension/activerecord/Model类的引用。
  6. #54 = Utf8 Lio/swagger/annotations/ApiModelProperty;:

    • 这是一个 UTF-8 编码的字符串,表示io.swagger.annotations.ApiModelProperty注解的类描述。
  7. #82 = Utf8 (Ljava/lang/Integer;)V:

    • 这是一个 UTF-8 编码的方法描述符,表示方法setId的参数为Integer,返回值为void
  8. #143 = Utf8 Student(id=:

    • 这是一个 UTF-8 编码的字符串,表示字符串"Student(id="。
  9. #165 = Utf8 append:

    • 这是一个 UTF-8 编码的字符串,表示方法append
  10. #168 = Class #168:

  • 这是一个类引用(Class)常量,表示对java/lang/Integer类的引用。

这只是常量池的一小部分,每个常量都有特定的类型和含义。常量池中的信息用于描述类的结构、字段、方法等。上述解析结果是对常量池中一些常见常量的简要说明。如果需要更详细的分析,可能需要查看整个常量池的内容。

Constant pool:#1 = Methodref          #35.#126      // com/baomidou/mybatisplus/extension/activerecord/Model."<init>":()V#2 = Fieldref           #7.#127       // com/kwan/springbootkwan/entity/Student.id:Ljava/lang/Integer;#3 = Fieldref           #7.#128       // com/kwan/springbootkwan/entity/Student.question:Ljava/lang/String;#4 = Fieldref           #7.#129       // com/kwan/springbootkwan/entity/Student.response:Ljava/lang/String;#5 = Fieldref           #7.#130       // com/kwan/springbootkwan/entity/Student.createTime:Ljava/util/Date;#6 = Fieldref           #7.#131       // com/kwan/springbootkwan/entity/Student.isDelete:Ljava/lang/Integer;#7 = Class              #132          // com/kwan/springbootkwan/entity/Student#8 = Methodref          #7.#133       // com/kwan/springbootkwan/entity/Student.canEqual:(Ljava/lang/Object;)Z#9 = Methodref          #7.#134       // com/kwan/springbootkwan/entity/Student.getId:()Ljava/lang/Integer;#10 = Methodref          #135.#136     // java/lang/Object.equals:(Ljava/lang/Object;)Z#11 = Methodref          #7.#137       // com/kwan/springbootkwan/entity/Student.getIsDelete:()Ljava/lang/Integer;#12 = Methodref          #7.#138       // com/kwan/springbootkwan/entity/Student.getQuestion:()Ljava/lang/String;#13 = Methodref          #7.#139       // com/kwan/springbootkwan/entity/Student.getResponse:()Ljava/lang/String;#14 = Methodref          #7.#140       // com/kwan/springbootkwan/entity/Student.getCreateTime:()Ljava/util/Date;#15 = Methodref          #135.#141     // java/lang/Object.hashCode:()I#16 = Class              #142          // java/lang/StringBuilder#17 = Methodref          #16.#126      // java/lang/StringBuilder."<init>":()V#18 = String             #143          // Student(id=#19 = Methodref          #16.#144      // java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;#20 = Methodref          #16.#145      // java/lang/StringBuilder.append:(Ljava/lang/Object;)Ljava/lang/StringBuilder;#21 = String             #146          // , question=#22 = String             #147          // , response=#23 = String             #148          // , createTime=#24 = String             #149          // , isDelete=#25 = String             #150          // )#26 = Methodref          #16.#151      // java/lang/StringBuilder.toString:()Ljava/lang/String;#27 = String             #152          // qinyingjie2#28 = Fieldref           #7.#153       // com/kwan/springbootkwan/entity/Student.notFinal:Ljava/lang/String;#29 = Methodref          #154.#155     // java/lang/Integer.valueOf:(I)Ljava/lang/Integer;#30 = Fieldref           #7.#156       // com/kwan/springbootkwan/entity/Student.num:Ljava/lang/Integer;#31 = Class              #157          // java/lang/String#32 = String             #158          // qinyuchen#33 = Methodref          #31.#159      // java/lang/String."<init>":(Ljava/lang/String;)V#34 = Fieldref           #7.#160       // com/kwan/springbootkwan/entity/Student.newStr:Ljava/lang/String;#35 = Class              #161          // com/baomidou/mybatisplus/extension/activerecord/Model#36 = Class              #162          // java/io/Serializable#37 = Utf8               serialVersionUID#38 = Utf8               J#39 = Utf8               ConstantValue#40 = Long               1l#42 = Utf8               name#43 = Utf8               Ljava/lang/String;#44 = String             #163          // qinyingjie1#45 = Utf8               notFinal#46 = Utf8               num#47 = Utf8               Ljava/lang/Integer;#48 = Utf8               intnum#49 = Utf8               I#50 = Integer            30#51 = Utf8               newStr#52 = Utf8               id#53 = Utf8               RuntimeVisibleAnnotations#54 = Utf8               Lio/swagger/annotations/ApiModelProperty;#55 = Utf8               value#56 = Utf8               主键id#57 = Utf8               question#58 = Utf8               问题#59 = Utf8               response#60 = Utf8               回答#61 = Utf8               createTime#62 = Utf8               Ljava/util/Date;#63 = Utf8               创建时间#64 = Utf8               isDelete#65 = Utf8               删除标识#66 = Utf8               <init>#67 = Utf8               ()V#68 = Utf8               Code#69 = Utf8               LineNumberTable#70 = Utf8               LocalVariableTable#71 = Utf8               this#72 = Utf8               Lcom/kwan/springbootkwan/entity/Student;#73 = Utf8               getId#74 = Utf8               ()Ljava/lang/Integer;#75 = Utf8               getQuestion#76 = Utf8               ()Ljava/lang/String;#77 = Utf8               getResponse#78 = Utf8               getCreateTime#79 = Utf8               ()Ljava/util/Date;#80 = Utf8               getIsDelete#81 = Utf8               setId#82 = Utf8               (Ljava/lang/Integer;)V#83 = Utf8               MethodParameters#84 = Utf8               setQuestion#85 = Utf8               (Ljava/lang/String;)V#86 = Utf8               setResponse#87 = Utf8               setCreateTime#88 = Utf8               (Ljava/util/Date;)V#89 = Utf8               setIsDelete#90 = Utf8               equals#91 = Utf8               (Ljava/lang/Object;)Z#92 = Utf8               o#93 = Utf8               Ljava/lang/Object;#94 = Utf8               other#95 = Utf8               this$id#96 = Utf8               other$id#97 = Utf8               this$isDelete#98 = Utf8               other$isDelete#99 = Utf8               this$question#100 = Utf8               other$question#101 = Utf8               this$response#102 = Utf8               other$response#103 = Utf8               this$createTime#104 = Utf8               other$createTime#105 = Utf8               StackMapTable#106 = Class              #132          // com/kwan/springbootkwan/entity/Student#107 = Class              #164          // java/lang/Object#108 = Utf8               canEqual#109 = Utf8               hashCode#110 = Utf8               ()I#111 = Utf8               PRIME#112 = Utf8               result#113 = Utf8               $id#114 = Utf8               $isDelete#115 = Utf8               $question#116 = Utf8               $response#117 = Utf8               $createTime#118 = Utf8               toString#119 = Utf8               <clinit>#120 = Utf8               Signature#121 = Utf8               Lcom/baomidou/mybatisplus/extension/activerecord/Model<Lcom/kwan/springbootkwan/entity/Student;>;Ljava/io/Serializable;#122 = Utf8               SourceFile#123 = Utf8               Student.java#124 = Utf8               Lio/swagger/annotations/ApiModel;#125 = Utf8               chatgpt问答实体类#126 = NameAndType        #66:#67       // "<init>":()V#127 = NameAndType        #52:#47       // id:Ljava/lang/Integer;#128 = NameAndType        #57:#43       // question:Ljava/lang/String;#129 = NameAndType        #59:#43       // response:Ljava/lang/String;#130 = NameAndType        #61:#62       // createTime:Ljava/util/Date;#131 = NameAndType        #64:#47       // isDelete:Ljava/lang/Integer;#132 = Utf8               com/kwan/springbootkwan/entity/Student#133 = NameAndType        #108:#91      // canEqual:(Ljava/lang/Object;)Z#134 = NameAndType        #73:#74       // getId:()Ljava/lang/Integer;#135 = Class              #164          // java/lang/Object#136 = NameAndType        #90:#91       // equals:(Ljava/lang/Object;)Z#137 = NameAndType        #80:#74       // getIsDelete:()Ljava/lang/Integer;#138 = NameAndType        #75:#76       // getQuestion:()Ljava/lang/String;#139 = NameAndType        #77:#76       // getResponse:()Ljava/lang/String;#140 = NameAndType        #78:#79       // getCreateTime:()Ljava/util/Date;#141 = NameAndType        #109:#110     // hashCode:()I#142 = Utf8               java/lang/StringBuilder#143 = Utf8               Student(id=#144 = NameAndType        #165:#166     // append:(Ljava/lang/String;)Ljava/lang/StringBuilder;#145 = NameAndType        #165:#167     // append:(Ljava/lang/Object;)Ljava/lang/StringBuilder;#146 = Utf8               , question=#147 = Utf8               , response=#148 = Utf8               , createTime=#149 = Utf8               , isDelete=#150 = Utf8               )#151 = NameAndType        #118:#76      // toString:()Ljava/lang/String;#152 = Utf8               qinyingjie2#153 = NameAndType        #45:#43       // notFinal:Ljava/lang/String;#154 = Class              #168          // java/lang/Integer#155 = NameAndType        #169:#170     // valueOf:(I)Ljava/lang/Integer;#156 = NameAndType        #46:#47       // num:Ljava/lang/Integer;#157 = Utf8               java/lang/String#158 = Utf8               qinyuchen#159 = NameAndType        #66:#85       // "<init>":(Ljava/lang/String;)V#160 = NameAndType        #51:#43       // newStr:Ljava/lang/String;#161 = Utf8               com/baomidou/mybatisplus/extension/activerecord/Model#162 = Utf8               java/io/Serializable#163 = Utf8               qinyingjie1#164 = Utf8               java/lang/Object#165 = Utf8               append#166 = Utf8               (Ljava/lang/String;)Ljava/lang/StringBuilder;#167 = Utf8               (Ljava/lang/Object;)Ljava/lang/StringBuilder;#168 = Utf8               java/lang/Integer#169 = Utf8               valueOf#170 = Utf8               (I)Ljava/lang/Integer;

5.canEqual

canEqual 方法通常用于实现 Java 中的 equals 方法。在实现类的 equals 方法时,通常需要遵循一些约定,其中之一是覆盖 equals 方法时也要覆盖 hashCode 方法。在实现 equals 方法时,需要比较两个对象的类型,以防止比较不同类型的对象。

canEqual 方法用于检查传入的对象是否是相同类的实例,这是为了确保在比较两个对象时不会因为类型不同而导致错误。通常,canEqual 方法的实现会使用 instanceof 运算符来进行类型检查。

protected boolean canEqual(final Object other) {return other instanceof Student;
}

6.构造方法说明

  • class 文件中方法会被{}括起来
  • Student()标识是 Student 类的构造函数
{public com.kwan.springbootkwan.entity.Student();descriptor: ()Vflags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: invokespecial #1                  // Method com/baomidou/mybatisplus/extension/activerecord/Model."<init>":()V4: returnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;
}

这是一个默认构造方法的字节码表示,对应于 com.kwan.springbootkwan.entity.Student 类的构造方法。下面是对该字节码的详细解释:

  • public com.kwan.springbootkwan.entity.Student();

    • 这是构造方法的声明,表示一个公共的无参数构造方法,属于 com.kwan.springbootkwan.entity.Student 类。
  • descriptor: ()V

    • 描述符表示方法的签名,这里 () 表示无参数,V 表示返回类型为 void
  • flags: ACC_PUBLIC

    • 表示该方法是公共的。
  • Code:

    • 下面是方法的字节码指令。
  • stack=1, locals=1, args_size=1

    • 这是关于操作数栈局部变量方法参数大小的信息。
    • 对于无参数方法,这三个值通常都是 1。
    • stack表示方法的操作数栈的最大深度,即在方法执行期间,最多同时存在多少个数值在操作数栈上。对于无参数方法,通常只需一个槽位来存储当前对象的引用(this)。
    • locals表示局部变量表的大小,即方法在执行期间可以使用的局部变量槽位数量。对于无参数方法,通常只需一个槽位来存储当前对象的引用(this)。
    • args_size表示方法的参数数量。对于无参数方法,参数数量为 1,即当前对象引用(this)。
  • 0: aload_0

    • 将当前对象引用(this)加载到操作数栈上。
    • 0 代表字节码中的第 0 条指令。偏移量是 0。
  • 1: invokespecial #1

    • 调用父类构造方法。invokespecial 是用于调用特殊方法(比如构造方法、私有方法)的指令。#1 是对常量池中索引为 1 的方法引用的引用。
    • 1 代表字节码中的第 1 条指令。偏移量是 1。
  • 4: return

    • 返回,结束方法执行。
    • 4 代表字节码中的第 1 条指令。偏移量是 4。
  • LineNumberTable:

    • 表示源代码行号信息。
  • line 12: 0

    • 在源代码的第 12 行,对应于字节码的第 0 指令。
  • LocalVariableTable:

    • 表示局部变量表信息。
  • Start Length Slot Name Signature

    • 描述局部变量表的开始位置、长度、槽位、变量名和变量签名。
  • 0 5 0 this Lcom/kwan/springbootkwan/entity/Student;

    • 表示在方法执行期间,槽位为 0 的局部变量是当前对象引用,变量名为 this,类型为 Lcom/kwan/springbootkwan/entity/Student;

总体而言,这段字节码表示 com.kwan.springbootkwan.entity.Student 类的公共无参数构造方法,其中通过 aload_0 指令加载当前对象引用到操作数栈上,然后通过 invokespecial 指令调用父类构造方法,最后通过 return 指令返回。

7.get 方法说明

getId 和构造函数方法过程类似,差异不大,可以类比分析,这里具体分析省略。

public java.lang.Integer getId();descriptor: ()Ljava/lang/Integer;flags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: getfield      #2                  // Field id:Ljava/lang/Integer;4: areturnLineNumberTable:line 28: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;

8.set 方法说明

public void setId(java.lang.Integer);descriptor: (Ljava/lang/Integer;)Vflags: ACC_PUBLICCode:stack=2, locals=2, args_size=20: aload_01: aload_12: putfield      #2                  // Field id:Ljava/lang/Integer;5: returnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       6     0  this   Lcom/kwan/springbootkwan/entity/Student;0       6     1    id   Ljava/lang/Integer;MethodParameters:Name                           Flagsid                             final

这段字节码对应于一个 Java 方法,该方法名为 setId,参数为一个 java.lang.Integer 类型的对象。下面是对这段字节码的详细解析:

  • public void setId(java.lang.Integer);

    • 这是方法的声明,表示一个公共的无返回值方法,名为 setId,接受一个 java.lang.Integer 类型的参数。
  • descriptor: (Ljava/lang/Integer;)V

    • 方法的描述符,指定了方法的签名。这里 (Ljava/lang/Integer;)V 表示方法接受一个 java.lang.Integer 类型的参数,返回类型为 void
  • flags: ACC_PUBLIC

    • 表示该方法是公共的。
  • Code:

    • 下面是方法的字节码指令。
  • stack=2, locals=2, args_size=2

    • stack=2 表示方法的操作数栈的最大深度是 2。
    • locals=2 表示局部变量表的大小是 2。
    • args_size=2 表示方法的参数数量是 2。
  • 0: aload_0

    • 将当前对象的引用(this)加载到操作数栈上。
  • 1: aload_1

    • 将方法参数 id 的值加载到操作数栈上。
  • 2: putfield #2

    • 将操作数栈顶的值(id)赋值给对象的字段。#2 表示常量池中索引为 2 的字段引用。
  • 5: return

    • 返回,结束方法执行。
  • LineNumberTable:

    • 表示源代码行号信息。
  • line 12: 0

    • 在源代码的第 12 行,对应于字节码的第 0 指令。
  • LocalVariableTable:

    • 表示局部变量表信息。
  • Start Length Slot Name Signature

    • 描述局部变量表的开始位置、长度、槽位、变量名和变量签名。
  • 0 6 0 this Lcom/kwan/springbootkwan/entity/Student;

    • 表示在方法执行期间,槽位为 0 的局部变量是当前对象引用,变量名为 this,类型为 Lcom/kwan/springbootkwan/entity/Student;
  • 0 6 1 id Ljava/lang/Integer;

    • 表示在方法执行期间,槽位为 1 的局部变量是参数 id,类型为 Ljava/lang/Integer;
  • MethodParameters:

    • 表示方法参数的信息。
  • Name Flags

    • 参数的名称和标志。
  • id final

    • 参数的名称为 id,并且标记为 final

9.常量方法

static {};descriptor: ()Vflags: ACC_STATICCode:stack=3, locals=0, args_size=00: ldc           #27                 // String qinyingjie22: putstatic     #28                 // Field notFinal:Ljava/lang/String;5: bipush        207: invokestatic  #29                 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;10: putstatic     #30                 // Field num:Ljava/lang/Integer;13: new           #31                 // class java/lang/String16: dup17: ldc           #32                 // String qinyuchen19: invokespecial #33                 // Method java/lang/String."<init>":(Ljava/lang/String;)V22: putstatic     #34                 // Field newStr:Ljava/lang/String;25: returnLineNumberTable:line 19: 0line 21: 5line 25: 13

10.操作数栈和局部变量表区别?

stack=1, locals=1, args_size=1

  • 这是关于操作数栈局部变量方法参数大小的信息。
  • 对于无参数方法,这三个值通常都是 1。
  • stack表示方法的操作数栈的最大深度,即在方法执行期间,最多同时存在多少个数值在操作数栈上。对于无参数方法,通常只需一个槽位来存储当前对象的引用(this)。
  • locals表示局部变量表的大小,即方法在执行期间可以使用的局部变量槽位数量。对于无参数方法,通常只需一个槽位来存储当前对象的引用(this)。
  • args_size表示方法的参数数量。对于无参数方法,参数数量为 1,即当前对象引用(this)。

操作数栈(Operand Stack)和局部变量表(Local Variable Table)是 Java 虚拟机(JVM)中两个关键的数据结构,用于支持方法的执行。它们在存储和管理数据的方式上有一些重要的区别:

操作数栈(Operand Stack):

  • 作用: 操作数栈是一个后进先出(LIFO)的栈,用于存储方法执行期间的临时数据中间结果。它是一种运算时的工作区,用于保存方法调用过程中产生的数据以及进行操作数的压栈弹栈操作。
  • 访问: JVM 的字节码指令通常包括对操作数栈的读写操作。例如,将数据推入操作数栈、弹出数据、进行算术运算等。
  • 生命周期: 操作数栈中的数据生命周期短暂,仅在方法的执行期间有效。当方法调用结束时,操作数栈的内容也被销毁。

局部变量表(Local Variable Table):

  • 作用: 局部变量表是一个数组,用于存储方法中的局部变量。局部变量包括方法参数临时变量以及在方法体内部定义其他变量。与操作数栈不同,局部变量表的数据在方法的整个生命周期内都是可访问的。
  • 访问: 局部变量表通过索引来访问,而不像操作数栈那样通过栈顶指针。变量的访问是直接的,通过索引来定位,而不需要像操作数栈那样进行栈顶的推入和弹出操作。
  • 生命周期: 局部变量表中的数据的生命周期长于操作数栈,它们在整个方法的执行过程中都可以被访问。

总的来说,操作数栈主要用于执行方法过程中的计算和操作,而局部变量表则用于存储方法中的局部变量。这两个数据结构协同工作,支持方法的正确执行。

11.方法表总结

  • 静态方法放在第一位,然后是构造方法,最后是其他方法
  • 方法声明,方法表用{}括起来,里面都是方法
  • 方法描述符和方法签名(descriptor)
  • 方法访问标志(flags)
  • 方法的字节码指令(code)
  • 方法的参数信息(MethodParameters)
{public com.kwan.springbootkwan.entity.Student();descriptor: ()Vflags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: invokespecial #1                  // Method com/baomidou/mybatisplus/extension/activerecord/Model."<init>":()V4: returnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;
}

12.方法的字节码指令

方法的字节码指令(code)包含如下内容

  • 操作数栈
  • 局部变量表
  • 方法参数大小
  • 源代码行号信息
  • 局部变量表的详细信息
  • 方法参数的详细信息
{public com.kwan.springbootkwan.entity.Student();descriptor: ()Vflags: ACC_PUBLICCode:stack=1, locals=1, args_size=10: aload_01: invokespecial #1                  // Method com/baomidou/mybatisplus/extension/activerecord/Model."<init>":()V4: returnLineNumberTable:line 12: 0LocalVariableTable:Start  Length  Slot  Name   Signature0       5     0  this   Lcom/kwan/springbootkwan/entity/Student;
}
  • Code:

    • 下面是方法的字节码指令。
  • stack=1, locals=1, args_size=1

    • 这是关于操作数栈局部变量方法参数大小的信息。
    • 对于无参数方法,这三个值通常都是 1。
    • stack表示方法的操作数栈的最大深度,即在方法执行期间,最多同时存在多少个数值在操作数栈上。对于无参数方法,通常只需一个槽位来存储当前对象的引用(this)。
    • locals表示局部变量表的大小,即方法在执行期间可以使用的局部变量槽位数量。对于无参数方法,通常只需一个槽位来存储当前对象的引用(this)。
    • args_size表示方法的参数数量。对于无参数方法,参数数量为 1,即当前对象引用(this)。
  • 0: aload_0

    • 将当前对象引用(this)加载到操作数栈上。
    • 0 代表字节码中的第 0 条指令。偏移量是 0。
  • 1: invokespecial #1

    • 调用父类构造方法。invokespecial 是用于调用特殊方法(比如构造方法、私有方法)的指令。#1 是对常量池中索引为 1 的方法引用的引用。
    • 1 代表字节码中的第 1 条指令。偏移量是 1。
  • 4: return

    • 返回,结束方法执行。
    • 4 代表字节码中的第 1 条指令。偏移量是 4。
  • LineNumberTable:

    • 表示源代码行号信息。
  • line 12: 0

    • 在源代码的第 12 行,对应于字节码的第 0 指令。
  • LocalVariableTable:

    • 表示局部变量表信息。
  • Start Length Slot Name Signature

    • 描述局部变量表的开始位置、长度、槽位、变量名和变量签名。
  • 0 5 0 this Lcom/kwan/springbootkwan/entity/Student;

    • 表示在方法执行期间,槽位为 0 的局部变量是当前对象引用,变量名为 this,类型为 Lcom/kwan/springbootkwan/entity/Student;

四.Class 文件结构

1.谈谈你对 class 文件的了解

类型名称数量
u4magic1
u2minor_version1
u2major_version1
u2constant_pool_count1
cp_infoconstant_poolconstant_pool_count - 1
u2access_flags1
u2this_class1
u2super_class1
u2interfaces_count1
u2interfacesinterfaces_count
u2fields_count1
field_infofieldsfields_count
u2methods_count1
method_infomethodsmethods_count
u2attributes_count1
attribute_infoattributesattributes_count

以上是 Java 类文件格式中的各个部分的名称、类型和数量的展示。其中:

  • u4 表示 4 个字节无符号整数
  • u2 表示 2 个字节无符号整数

constant_pool_count: 表示常量池中的常量数量(不包括占用两个位置的 long 和 double 类型常量,所以要减去 1)

在 Java 类文件格式中,常量池是一个重要的部分,它包含了类、接口、字段、方法等信息的符号引用和字面量常量。其他部分则包括类版本信息、访问标志、接口列表、字段表、方法表和属性表等。

2.fields 和 attributes 区别

在 Java 的 Class 文件中,fields(字段表)和attributes(属性表)是两个不同的部分,用于描述类或接口的结构和特性。

  1. Fields(字段表):
    • fields部分用于描述接口中定义的字段(成员变量)信息。
    • 每个字段都包含了字段的访问修饰符、字段名称、字段类型等信息。
    • 字段表中记录了类或接口中所有的字段,无论是静态字段还是实例字段,包括公共的、私有的、保护的和默认访问权限的字段。
  2. Attributes(属性表):
    • attributes部分用于描述字段方法代码等部分的额外属性信息。
    • 属性表中包含了各种不同类型的属性,这些属性可以用于存储额外的元数据,供 Java 虚拟机和其他工具使用。
    • 在类级别,属性表中可能包含SourceFile(源文件名)、InnerClasses(内部类信息)、EnclosingMethod(外部类和方法信息)等属性。
    • 在字段级别,属性表中可能包含ConstantValue(常量值)、Synthetic(合成字段标记)等属性。
    • 在方法级别,属性表中可能包含Code(字节码)、Exceptions(异常表)、LineNumberTable(行号表)等属性。

总结: fields部分记录了类或接口中定义的字段信息,而attributes部分用于描述类、字段、方法或代码等部分的额外属性信息。fields主要描述类的结构,而attributes主要用于存储附加的元数据和信息,用于支持 Java 虚拟机和其他工具的功能。

3.DemoTest 解析

class 文件是以一组 8 个字节为基础单位的二进制流,各个数据项严格按照顺序紧凑排列在文件中,中间没有任何分隔符,这使得 class 文件存储的都是程序运行的必要数据,没有空隙存在。

class 文件格式采用一种类似 c 语言结构体的伪结构体老存储数据,这种伪结构体只包含 2 种数据类型:无符号数和表。

无符号数属于基本的数据类型,以 u1,u2,u4,u8 分别来表示 1 个字节,2 个字节,4 个字节,8 个字节的无符号数.无符号数可以用来描述数字,索引引用,数量值或者按照 utf-8 编码构成的字符串值。

表是由多个无符号数或者其他表作为数据项构成的复合型数据结构,为了便于区分,表通常以_info 结尾.用于描述复杂的结构,整个 class 文件可以看成一个大表。

image-20230727204139856

ClassFile {u4             magic;//魔数u2             minor_version;//次版本号u2             major_version;//主版本号u2             constant_pool_count;//常量池数量cp_info        constant_pool[constant_pool_count-1];//常量池信息u2             access_flags;//访问标志u2             this_class;//类索引u2             super_class;//父类索引u2             interfaces_count;//接口数(2位,所以一个类最多65535个接口)u2             interfaces[interfaces_count];//接口索引u2             fields_count;//字段数field_info     fields[fields_count];//字段表集合u2             methods_count;//方法数method_info    methods[methods_count];//方法集合u2             attributes_count;//属性数attribute_info attributes[attributes_count];//属性表集合
}

4.class 文件中的魔数和主次版本号?

每个 class 文件的头 4 个字节被称为魔数,它的唯一作用是确定这个文件是否能为一个虚拟机所接受的 class 文件。紧接着是 class 文件的版本号,第五和第六字节是次版本号,第七第八是主版本号,java 的版本号是从 45 开始的,jdk1.1 开始每个 jdk 大版本发布主版本号向上加 1。

image-20230727210121435

5.为什么常量池计数器从 1 开始

由于常量池中常量的数量是不固定的,所以常量池中的入口需要放置一项 u2 类型的数据,代表常量池容量计数器,constant_pool_count,这个计数器是从 1 开始的不是从 0 开始的,如下图所示,十六进制数 0x0016,十进制就是 22,代表着常量池中有 21 项常量,索引范围为 1~21。

第 0 项表示不引用任何一个常量池项目.class 文件只有常量池的容量是从 1 开始的,对于其他的集合类型,包括接口索引集合,字段表集合,方法表集合等的容量计数器都是从 0 开始的。

  • 匿名内部类本身没有类名称,进行名称引用时,会将 index 指向 0
  • Object 类的 class 文件父类索引指向 0

image-20230727210145421

6.class 文件常量池中存放的是什么内容?

常量池中主要存放两大类常量:字面量和符号引用

字面量比较接近于 java 语言层面的常量概念,如文本字符串,被声明为 final 的常量值等

符号引用则属于偏编译方面的概念,主要包含以下几类常量:

  • 类和接口的全限定名
  • 字段的名称和描述符
  • 方法的名称和描述符
  • 方法句柄和方法类型
  • 动态调用点和动态常量

image-20230727210348919

7.常量池的项目类型?

十四种常量都有自己的结构.

image-20231022231003264

CONSTANT_Class_info: 代表类或接口的符号引用.数据结构如下.

tag 标志位用于区分常量类型,name_index 是一个索引值,指向常量池中一个 CONSTANT_Utf8_info 类型的常量.此常量代表了这个类或接口的全限定名.

image-20231020114029553

CONSTANT_Utf8_info: 的数据结构如下

  • tag 是标志位区分类型

  • length 代表数据长度,最大为 65535

  • bytes 代表实际数据

image-20231022231024181

常量项目类型描述
CONSTANT_Utf8_infotagu1值为 1
lengthu2UTF-8 编码的字符串占用的字节数
bytesu1长度为 length 的 UTF-8 编码的字符串
CONSTANT_Integer_infotagu1值为 3
bytesu4按照高位在前存储的 int 值
CONSTANT_Float_infotagu1值为 4
bytesu4按照高位在前存储的 float 值
CONSTANT_Long_infotagu5值为 5
bytesu8按照高位在前存储的 long 值
CONSTANT_Double_infotagu1值为 6
bytesu8按照高位在前存储的 double 值
CONSTANT_Class_infotagu1值为 7
indexu2指向全限定名常量项的索引
CONSTANT_String_infotagu1值为 8
indexu2指向字符串字面量的索引
CONSTANT_Fieldref_infotagu1值为 9
indexu2指向声明字段的类或者接口描述符 CONSTANT_Class_info 的索引项
indexu2指向字段描述符 CONSTANT_NameAndType 的索引项
CONSTANT_Methodref_infotagu1值为 10
indexu2指向声明方法的类描述符 CONSTANT_ Class_info 的索引项
indexu2指向名称及类型描述符 CONSTANT_NameAndType 的索引项
CONSTANT_Interface_Methodref _infotagu1值为 11
indexu2指向声明方法的接口描述符 CONSTANT_Class_info 的索引项
indexu2指向名称及类型描述符 CONSTANT_NameAndType 的索引项
CONSTANT_NameAndType_infotagu1值为 12
indexu2指向该字段或方法名称常量项的索引
indexu2指向该字段或方法描述符常量项的索引
CONSTANT_MethodHandle_infotagu1值为 15
reference_kindu1值必须在 1~9 之间(包括 1 和 9)它决定了方法句柄的类型,方法句柄类型的值表示方法句柄的字节码行为
reference_indexu2值必须是对常量池的有效索引
CONSTANT_MethodType_infotagu1值为 16
descriptor_indexu2值必须是对常量池的有效索引,常量池在该索引处的项必须是 CONSTANT Utf8info 结构,表示方法的描述符
CONSTANT_Invoke_Dynamic_infotagu1值为 18
bootstrap_method_attr_indexu2值必须是对当前 Class 文件中引导方法表的 bootstrap_methods 数组的有效索引
name_and_type_indexu2值必须是对当前常量池的有效索引,常量池在该索引处的项必须是 CONSTANT_NameAndType_info 结构,表示方法名和方法描述符

8.java 字段名和方法名长度限制?

字段名和方法名都存储在常量池中

存储这 2 个名称需要用到常量池中的 constant_utf8_info 类型来存储,以下是 constant_utf8_info 的存储结构,

image-20231022231037848

constant_utf8_info 的最大长度也是 java 中方法名字段名的长度,这里最大长度就是 length 的最大值,即 u2 类型能表达的最大值为 65535,所以 java 程序中定义了超过 64kb 英文字符的变量或者方法名,即使规则和名字符号全部合法,也无法编译。

  • 1kb=1024 字节
  • 64kb=65536 字节是临街值,不能等于 64kb

9.class 文件的访问标志作用?

常量池结束后,是 2 个字节的访问标志,用于标示类和接口层次的访问信息.包括这个 class 是类还是接口,是否定义为 public 类型,是否定义为 abstract 类型,如果是类的话,是否声明为 final 等等。

image-20230727210842600

image-20230727210859601

10.类索引、父类索引、接口索引集合作用

类索引(this_class)和父类索引(super_class)都是一个 u2 类型的数据。而接口索引集合(interfaces)是一组 u2 类型的数据集合,class 文件中由这三项数据来确定该类的继承实现关系。

image-20230727211041376

类索引用于确定这个类的全限定名(全限定名称存储于常量池),父类索引用于确定这个类的父类的全限定名.这里说的索引,是指向常量池中的 constant_class_info 类型,constant_class_info 又指向了 constant_utf8_info 从而确定全限定名。

由于 java 语言不支持多继承,所以父类索引只有一个,除了 java.lang.Object 之外,所有的 java 类都有父类,且父类索引都不为 0,接口索引集合就是用来描述这个类实现了哪些接口,这些被实现的接口将按 implements 关键字后的接口顺序从左到右排列在接口索引集合中。

类索引,父类索引,接口索引集合都按顺序排列在访问标志之后,类索引和父类索引,用 2 个 u2 类型的值表示,他们各自指向一个 constant_class_info 的类描述符常量,通过 constant_class_info 类型的常量中的索引值可以找到定义在 constant_utf8_info 类型的常量中的全限定名字符串。

image-20231022231054267

11.class 文件的字段表存储的什么信息?

image-20231022231120789

字段表(field_info)用于描述接口或者类中声明的变量,java 语言中的字段包括类级变量以及实例级变量,但不包含在方法内部声明的局部变量。

字段表存储的是变量的修饰符+字段的描述符索引(索引指向常量池)+字段名称索引(索引指向常量池).

修饰符:字段可以包括的修饰符有字段的作用域(public,private,protected 修饰符),是实例变量还是类变量(static 修饰符),可变性(final),并发可见性(volatile 修饰符,是否强制从主内存读写),可否被序列化(transient 修饰符)等。

描述符:字段类型。

public final static String number=“1”,public final 和 static 是访问修饰符access_flags,这些都存在 class 文件的字段表中,String 是字段描述符,存放于常量池中的 name_index,number 是字段的名称 descrip_index,存放于常量池,这两部分的关联,是通过字段表的 name_index 指向常量池中的字段名称 numberdescrip_index 指向常量池中的描述符。

image-20230727211618977

12.class 文件的方法表存储的什么?

class 文件存储格式中对方法的描述,采用的方式和字段表一致,方法表的结构和字段表一致,包含访问标示access_flags,名称索引name_index,描述符索引descriptor_index,属性表集合attributes

image-20230727211733376

image-20230727211747384

13.class 文件中的属性表?

  • ConstantValue 是在字段表下使用的;
  • Code 属性是在方法表下使用的;

image-20230727211915669

属性表在 class 文件字段表方法表都有自己的属性表集合,以描述某些场景专有的信息。与 class 文件中其他数据项要求严格的顺序,长度和内容不同,属性表限制稍微宽松一些,不再要求严格的顺序。只要属性名不重复,允许写入自己定义的属性信息。

对于每一个属性,它的名称都是从常量池中引用一个 constant_utf8_info 类型的常量来表示,而属性值的结构是完全自定义的,只需要通过一个 u4 的长度属性去说明属性值所占用的字节的位数即可。

image-20231022231142832

Code 属性:java 代码经过 javac 编译之后,最终变成字节码指令存储在 code 属性中。code 属性出现在方法表的属性集合中。但并非所有的方法表都必须存在这个属性,譬如接口和抽象类的方法就不存在 code 属性,如果方法表有 code 属性存在,那么它的结构如下:

image-20230727212132743

LineNumberTable 属性:LineNumberTable 属性用于描述 java 源码的行号和字节码行号之间的偏移量的对应关系。它不是运行时必需的属性,但默认会生成到 class 文件之中,可以在 javac 中使用-g:none 或者-g:lines 选项来取消或者要求生成这项信息,如果选择不生成 LineNumberTable 属性,对程序的影响是,抛出异常时,堆栈中将不显示出错的行号,并且在调试程序的时候,也无法按照源码行来设置断点。

image-20230727212323021

ConstantValue 属性:主要作用是为静态变量赋值,只有被 static 修饰的变量才能使用这个属性。int x=123 和 static int x=123 这样的定义在 java 中很常见,但虚拟机对这 2 种方式的赋值的方式和时刻有所不同。对于非 static 修饰的变量,在实例构造器 init 方法中进行赋值。对于类变量,有 2 种方式,通过类构造器的 clinit 方法赋值或者使用 ConstantValue 属性。如果同时使用 final 和 static 关键字修饰同一个变量,并且变量类型是基本数据类型或者 String,就会使用 ConstantValue 属性来进行初始化,如果没有用 final 修饰,或者非基本类型或者字符串,会在构造器 clinit 方法中初始化。

//举例说明
public static final String NUMBER=111111;

static 和 final 同时修饰的变量,会在编译的过程中把 NUBER 的值进行初始化,放到 ConstanValue 属性中,否则需要到最后一步初始化时在 clinit 方法中进行初始化。

image-20230727212648804

觉得有用的话点个赞 👍🏻 呗。
❤️❤️❤️本人水平有限,如有纰漏,欢迎各位大佬评论批评指正!😄😄😄

💘💘💘如果觉得这篇文对你有帮助的话,也请给个点赞、收藏下吧,非常感谢!👍 👍 👍

🔥🔥🔥Stay Hungry Stay Foolish 道阻且长,行则将至,让我们一起加油吧!🌙🌙🌙

img

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/195202.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

终端安全管理软件是监控软件吗

有些人在后台问&#xff0c;终端安全管理软件是监控软件吗&#xff1f; 先回答&#xff0c;是监控软件。 因为它具有监控的功能&#xff0c;在很大程度上&#xff0c;是可以用来当做监控软件来用的。 终端安全管理软件是一种集中管理终端设备的软件工具&#xff0c;可以在企业…

猜数字赢金币

充值金币后开始游戏&#xff0c;猜中奖励10金币退出&#xff0c;不中扣除1金币继续。 (笔记模板由python脚本于2023年12月03日 21:52:23创建&#xff0c;本篇笔记适合熟悉程序函数式编程&#xff0c;熟练掌握基本数据类型的coder翻阅) 【学习的细节是欢悦的历程】 Python 官网&…

linux常用命令-find命令与scp命令详解(超详细)

文章目录 前言一、find命令介绍1. find命令简介2. find命令的基本语法3. 常用的find命令选项和表达式 二、find命令示例用法1. 按照名称进行搜索2. 按照类型进行搜索3. 按照修改时间进行搜索4. 按照文件大小进行搜索5. 对搜索到的文件执行指定的命令6. 删除搜索到的文件 三、sc…

线程池怎么用?---实例讲解

线程池使用实例 先写一个配置类 /*** 线程池配置*/ Configuration public class ThreadPoolConfig {//定义线程前缀public static final String NAME_PRE"test";/*** ExecutorService 这个对象就是线程池&#xff0c;可以点进去他的源码看看* Bean&#xff0c;将ge…

2024年甘肃省职业院校技能大赛(中职教师组)网络安全竞赛样题卷③

2024年甘肃省职业院校技能大赛&#xff08;中职教师组&#xff09;网络安全竞赛样题卷③ 2024年甘肃省职业院校技能大赛&#xff08;中职教师组&#xff09;网络安全竞赛样题卷③A模块基础设施设置/安全加固&#xff08;200分&#xff09;A-1任务一 登录安全加固&#xff08;Wi…

400页Python学习PDF笔记,全面总结零基础入门看这一篇足够了

我们都知道Python入门比较简单&#xff0c;但仍有很多想要学习的新手依然卡在基础的安装阶段&#xff0c;尽管如此&#xff0c;网络上的大部分的教程却对这些基础内容都是一带而过&#xff0c;导致许多新手朋友对这些基础知识一知半解&#xff0c;往往一个小问题都需要在网上查…

电脑回收站还原的文件在哪里找到?如何找回回收站还原的文件

电脑回收站是一种非常有用的功能&#xff0c;可以帮助我们恢复无意中删除的文件。然而&#xff0c;许多人可能不清楚还原的文件在哪里可以找到。本文将为您带来详细解答&#xff0c;并帮助您找回回收站还原的文件。 电脑回收站还原的文件在哪里找到 当我们使用电脑的回收站功…

element的el-date-picker时间控件,限制选择范围区间天数并且当前之后的日期不可选

element的el-date-picker时间控件&#xff0c;限制选择范围区间天数并且当前之后的日期不可选 HTML部分代码 <el-date-pickerv-model"dateRange"type"datetimerange"value-format"yyyy-MM-dd HH:mm:ss"range-separator"至"start-p…

element 弹窗在弹出后鼠标还可以点击页面其他元素

文章目录 需求分析需求 如下图所示,在点击弹出弹框后,支持 鼠标可点击弹框外的其他地方可拖拽弹框弹出弹出后不可有遮挡弹出样式可自定义 分析 官网:https://vxetable.cn/v4/#/table/start/install 安装 vxe-table 引入import {App, createApp }

速达软件全系产品任意文件上传漏洞复现 [附POC]

文章目录 速达软件全系产品任意文件上传漏洞复现 [附POC]0x01 前言0x02 漏洞描述0x03 影响版本0x04 漏洞环境0x05 漏洞复现1.访问漏洞环境2.构造POC3.复现 0x06 修复建议 速达软件全系产品任意文件上传漏洞复现 [附POC] 0x01 前言 免责声明&#xff1a;请勿利用文章内的相关技…

centos安装Python3之后yum不能使用异常

场景&#xff1a; 需要在centos上安装Python3&#xff0c;但是安装Python3之后出现yum不能使用的问题。 问题描述 在centos上安装python3之后出现yum使用不了问题&#xff0c;使用yum会报如下信息&#xff1a; [roothadoop101~]# yum install wgetFile "/usr/bin/yum&q…

旋转框(obb)目标检测计算iou的方法

首先先定义一组多边形&#xff0c;这里的数据来自前后帧的检测结果 pre [[[860.0, 374.0], [823.38, 435.23], [716.38, 371.23], [753.0, 310.0]],[[829.0, 465.0], [826.22, 544.01], [684.0, 539.0], [686.78, 459.99]],[[885.72, 574.95], [891.0, 648.0], [725.0, 660.0]…

Matlab论文插图绘制模板第129期—函数网格曲面图

在之前的文章中&#xff0c;分享了Matlab函数折线图的绘制模板&#xff1a; 函数三维折线图&#xff1a; 进一步&#xff0c;再来分享一下函数网格曲面图。 先来看一下成品效果&#xff1a; 特别提示&#xff1a;本期内容『数据代码』已上传资源群中&#xff0c;加群的朋友请自…

activemq启动成功但web管理页面却无法访问

前提&#xff1a; 在linux启动activemq成功&#xff01;本地能ping通linux 处理方案&#xff1a; 确定防火墙是否关闭&#xff0c; 有两种处理方案&#xff1a;第一种-关闭防火墙&#xff1b;第二种-暴漏8161和61616两个端口 netstat -lnpt查看8161和61616端口 注意&#xf…

网络细节核心笔记

来源&#xff0c;做个笔记&#xff0c;讲的还蛮清楚通信原理-2.5 数据封装与传输05_哔哩哔哩_bilibili 交换机

慈善始于心,行善贵有恒 | 中创开展“寒冬送温暖”公益活动

岁暮隆冬&#xff0c;冷霜挂睫&#xff0c;前往尖山村的路上雾气弥漫&#xff0c;弯弯绕绕的山路需要开车一小时才能到达目的地。对许多人来说&#xff0c;这或许是一段漫长而艰辛的路程&#xff0c;但对于那些生活在山区的贫困儿童而言&#xff0c;这条山路却是通往外界的唯一…

IEEE 机器人最优控制开源库 Model-based Optimization for Robotics

系列文章目录 文章目录 系列文章目录前言一、开源的库和工具箱1.1 ACADO1.2 CasADi1.3 Control Toolbox1.4 Crocoddyl1.5 Ipopt1.6 Manopt1.7 LexLS1.8 NLOpt1.9 qpOASES1.10 qpSWIFT1.11 Roboptim 二、其他库和工具箱2.1 MUSCOD2.2 OCPID-DAE12.3 SNOPT 前言 机器人&#xff…

python爬虫基础html内容解析库BeautifulSoup

我们通过Requests请求url获取数据&#xff0c;请求把数据返回来之后就要提取目标数据&#xff0c;不同的网站返回的内容通常有多种不同的格式&#xff0c;一种是 json 格式&#xff0c;我们可以直接通过json.loads转换python的json对象处理。另一种 XML 格式的&#xff0c;还有…

【一个超简单的爬虫demo】探索新浪网:使用 Python 爬虫获取动态网页数据

探索新浪网&#xff1a;使用 Python 爬虫获取动态网页数据 引言准备工作选择目标新浪网的结构 编写爬虫代码爬取example.com爬取新浪首页部分内容解析代码注意&#xff1a; KeyError: href结果与展示 其他修改和适应注意事项 总结 引言 可以实战教爬虫吗&#xff0c;搭个环境尝…

快考个PMP,救救脆皮项目经理吧

最近&#xff0c;脆皮大学生在互联网上狠狠火了一把&#xff0c;各种稀奇古怪导致受伤的原因&#xff0c;各种意外骨折骨裂的方式&#xff0c;不断地刷新着网友的世界观。但喜番仔细思考&#xff0c;发现脆皮的不只是大学生&#xff0c;广大的打工人&#xff0c;特别是项目经理…