Java class文件格式之数据类型_动力节点Java学院整理

 更新时间:2017年06月14日 15:10:31   投稿:mrr  
这篇文章主要介绍了Java class文件格式之数据类型的相关资料,需要的朋友可以参考下

CONSTANT_Integer_info

一个常量池中的CONSTANT_Integer_info数据项, 可以看做是CONSTANT_Integer类型的一个实例。 它存储的是源文件中出现的int型数据的值。 同样, 作为常量池中的一种数据类型, 它的第一个字节也是一个tag值, 它的tag值为3, 也就是说, 当虚拟机读到一个tag值为3的数据项时, 就知道这个数据项是一个CONSTANT_Integer_info, 它存储的是int型数值的值。 紧挨着tag的下面4个字节叫做bytes, 就是int型数值的整型值。 它的内存布局如下:

下面以示例代码进行说明, 示例代码如下:

package com.bjpowernode.test; 
public class TestInt { 
 void printInt(){ 
  System.out.println(65535); 
 } 
} 

将上面的类生成的class文件反编译:

1.D:\Workspace\AndroidWorkspace\BlogTest\bin>javap -v -c -classpath . com.bjpowernode.test.TestInt  

下面列出反编译的结果, 由于反编译结果较长, 我们省略了大部分信息:

.................. 
 .................. 
Constant pool: 
 .................. 
 .................. 
 #21 = Integer   65535 
 .................. 
 .................. 
{ 
  .................. 
  .................. 
 void printInt(); 
 flags: 
 Code: 
  stack=2, locals=1, args_size=1 
   0: getstatic  #15     // Field java/lang/System.out:Ljava/io/PrintStream; 
   3: ldc   #21     // int 65535 
   5: invokevirtual #22     // Method java/io/PrintStream.println:(I)V 
   8: return 
  LineNumberTable: 
  line 6: 0 
  line 7: 8 
  LocalVariableTable: 
  Start Length Slot Name Signature 
    0  9  0 this Lcom/bjpowernode/test/TestInt; 
} 

上面的输出结果中, 保留了printInt方法的反编译结果, 并且保留了常量池中的第21项。 首先看printInt方法反编译结果中的索引为3 的字节码指令:

1.3: ldc           #21                 // int 65535  

这条ldc指令, 引用了常量池中的第21项, 而第21项是一个CONSTANT_Integer_info, 并且这个CONSTANT_Integer_info存储的整型值为65535 。 

CONSTANT_Float_info

一个常量池中的CONSTANT_Float_info数据项, 可以看做是CONSTANT_Float类型的一个实例。 它存储的是源文件中出现的float型数据的值。 同样, 作为常量池中的一种数据类型, 它的第一个字节也是一个tag值, 它的tag值为4, 也就是说, 当虚拟机读到一个tag值为4的数据项时, 就知道这个数据项是一个CONSTANT_Float_info, 并且知道它存储的是float型数值。 紧挨着tag的下面4个字节叫做bytes, 就是float型的数值。 它的内存布局如下:

举例说明, 如果源文件中的一句代码使用了一个float值, 如下所示:

void printFloat(){ 
 System.out.println(1234.5f); 
} 

那么在这个类的常量池中就会有一个CONSTANT_Float_info与之相对应, 这个CONSTANT_Float_info的形式如下:

代码反编译结果如下:

Constant pool: 
............ 
............ 
 #29 = Float    1234.5f 
........... 
........... 
{ 
............ 
............ 
 void printFloat(); 
 flags: 
 Code: 
  stack=2, locals=1, args_size=1 
   0: getstatic  #15     // Field java/lang/System.out:Ljava/io/PrintStream; 
   3: ldc   #29     // float 1234.5f 
   5: invokevirtual #30     // Method java/io/PrintStream.println:(F)V 
   8: return 
  LineNumberTable: 
  line 10: 0 
  line 11: 8 
  LocalVariableTable: 
  Start Length Slot Name Signature 
    0  9  0 this Lcom/bjpowernode/test/TestInt; 
} 

CONSTANT_Long_info

一个常量池中的CONSTANT_Long_info数据项, 可以看做是CONSTANT_Long类型的一个实例。 它存储的是源文件中出现的long型数据的值。 同样, 作为常量池中的一种数据类型, 它的第一个字节也是一个tag值, 它的tag值为5, 也就是说, 当虚拟机读到一个tag值为5的数据项时, 就知道这个数据项是一个CONSTANT_Long_info, 并且知道它存储的是long型数值。 紧挨着tag的下面8个字节叫做bytes, 就是long型的数值。 它的内存布局如下:

举例说明, 如果源文件中的一句代码使用了一个long型的数值, 如下所示:

void printLong(){ 
 System.out.println(123456L); 
} 

那么在这个类的常量池中就会有一个CONSTANT_Long_info与之相对应, 这个CONSTANT_Long_info的形式如下:

代码反编译结果为:

Constant pool: 
............. 
............. 
 #21 = Long    123456l 
............. 
............. 
{ 
.............. 
.............. 
 void printLong(); 
 flags: 
 Code: 
  stack=3, locals=1, args_size=1 
   0: getstatic  #15     // Field java/lang/System.out:Ljava/io/PrintStream; 
   3: ldc2_w  #21     // long 123456l 
   6: invokevirtual #23     // Method java/io/PrintStream.println:(J)V 
   9: return 
  LineNumberTable: 
  line 7: 0 
  line 8: 9 
  LocalVariableTable: 
  Start Length Slot Name Signature 
    0  10  0 this Lcom/bjpowernode/test/TestInt; 
} 

CONSTANT_Double_info

一个常量池中的CONSTANT_Double_info数据项, 可以看做是CONSTANT_Double类型的一个实例。 它存储的是源文件中出现的double型数据的值。 同样, 作为常量池中的一种数据类型, 它的第一个字节也是一个tag值, 它的tag值为6, 也就是说, 当虚拟机读到一个tag值为6的数据项时, 就知道这个数据项是一个CONSTANT_Double_info, 并且知道它存储的是double型数值。 紧挨着tag的下面8个字节叫做bytes, 就是double型的数值。 它的内存布局如下:


举例说明, 如果源文件中的一句代码使用了一个double型的数值, 如下所示:

void printDouble(){ 
 System.out.println(123456D); 
} 

那么在这个类的常量池中就会有一个CONSTANT_Double_info与之相对应, 这个CONSTANT_Double_info的形式如下:

代码反编译结果为:

Constant pool: 
............ 
............ 
 #21 = Double    123456.0d 
............ 
............ 
{ 
............. 
............. 
 void printDouble(); 
 flags: 
 Code: 
  stack=3, locals=1, args_size=1 
   0: getstatic  #15     // Field java/lang/System.out:Ljava/io/PrintStream; 
   3: ldc2_w  #21     // double 123456.0d 
   6: invokevirtual #23     // Method java/io/PrintStream.println:(D)V 
   9: return 
  LineNumberTable: 
  line 7: 0 
  line 8: 9 
  LocalVariableTable: 
  Start Length Slot Name Signature 
    0  10  0 this Lcom/bjpowernode/test/TestInt; 
} 

CONSTANT_String_info

在常量池中, 一个CONSTANT_String_info数据项, 是CONSTANT_String类型的一个实例。 它的作用是存储文字字符串, 可以把他看做是一个存在于class文件中的字符串对象。 同样, 它的第一个字节是tag值, 值为8 , 也就是说, 虚拟机访问一个数据项时, 判断tag值为8 , 就说明访问的数据项是一个CONSTANT_String_info 。 紧挨着tag的后两个字节是一个叫做string_index的常量池引用, 它指向一个CONSTANT_Utf8_info, 这个CONSTANT_Utf8_info存放的才是字符串的字面量。 它的内存布局如下:

 

举例说明, 如果源文件中的一句代码使用了一个字符串常量, 如下所示:

void printStrng(){ 
 System.out.println("abcdef"); 
} 

那么在这个类的常量池中就会有一个CONSTANT_String_info与之相对应, 反编译结果如下:

Constant pool: 
............ 
............ 
 #21 = String    #22   // abcdef 
 #22 = Utf8    abcdef 
............ 
............. 
{ 
............. 
............. 
 void printStrng(); 
 flags: 
 Code: 
  stack=2, locals=1, args_size=1 
   0: getstatic  #15     // Field java/lang/System.out:Ljava/io/PrintStream; 
   3: ldc   #21     // String abcdef 
   5: invokevirtual #23     // Method java/io/PrintStream.println:(Ljava/lang/String;)V 
   8: return 
  LineNumberTable: 
  line 7: 0 
  line 8: 8 
  LocalVariableTable: 
  Start Length Slot Name Signature 
    0  9  0 this Lcom/bjpowernode/test/TestInt; 
} 

其中printString方法中索引为3的字节码指令ldc引用常量池中的第21项, 第21项是一个CONSTANT_String_info, 这个位于第21项的CONSTANT_String_info又引用了常量池的第22项, 第22项是一个CONSTANT_Utf8_info, 这个CONSTANT_Utf8_info中存储的字符串是 abcdef 。 引用关系的内存布局如下:

 

总结

最后总结一下, 本文主要讲解了常量池中的五中数据项, 分别为CONSTANT_Integer_info, CONSTANT_Float_info, CONSTANT_Long_info, CONSTANT_Double_info 和CONSTANT_String_info 。 这几种常量池数据项都是直接存储的常量值,而不是符号引用。 这里又一次出现了符号引用的概念, 这个概念将会在下一篇博客中详细讲解, 因为下一篇博客要介绍的剩下的四种常量池数据项, 都是符号引用, 这四种表示符号引用的数据项又会直接或间接引用上篇文章中介绍的CONSTANT_NameAndType_info和CONSTANT_Utf8_info, 所以说CONSTANT_NameAndType_info是符号引用的一部分。

从本文中我们还可以知道。 虽然说CONSTANT_String_info是直接存储值的数据项, 但是CONSTANT_String_info有点特别, 因为它不是直接存储字符串, 而是引用了一个CONSTANT_Utf8_info, 这个被引用的CONSTANT_Utf8_info中存储了字符串。

相关文章

  • mybatis-plus中更新null值的问题解决

    mybatis-plus中更新null值的问题解决

    本文主要介绍 mybatis-plus 中常使用的 update 相关方法的区别,以及更新 null 的方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-04-04
  • java读取XML文件的四种方法总结(必看篇)

    java读取XML文件的四种方法总结(必看篇)

    下面小编就为大家带来一篇java读取XML文件的四种方法总结(必看篇)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-06-06
  • java使用筛选法求n以内的素数示例(java求素数)

    java使用筛选法求n以内的素数示例(java求素数)

    这篇文章主要介绍了java使用筛选法求n以内的素数示例(java求素数),需要的朋友可以参考下
    2014-04-04
  • Java编程将汉字转Unicode码代码示例

    Java编程将汉字转Unicode码代码示例

    偶然看到Unicode编码,觉得挺有意思,于是搜索了相关资料,准备学习学习,本文主要是一个Unicode编码的简单Java实现,需要的朋友可以了解下。
    2017-10-10
  • Java HashMap源码深入分析讲解

    Java HashMap源码深入分析讲解

    在java开发中,HashMap是最常用、最常见的集合容器类之一,下面一起温故一下,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-08-08
  • springcloud如何配置文件加载顺序

    springcloud如何配置文件加载顺序

    这篇文章主要介绍了springcloud如何配置文件加载顺序问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-07-07
  • Spring bean不被GC的真正原因及分析

    Spring bean不被GC的真正原因及分析

    这篇文章主要介绍了Spring bean不被GC的真正原因及分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-04-04
  • 浅谈Java中浮点型数据保留两位小数的四种方法

    浅谈Java中浮点型数据保留两位小数的四种方法

    今天在进行开发的过程中遇到了一个小问题,是关于如何将double类型的数据保留两位小数。具有一定的参考价值,本文就详细的介绍一下
    2021-09-09
  • SpringSecurity实现图形验证码功能的实例代码

    SpringSecurity实现图形验证码功能的实例代码

    Spring Security 的前身是 Acegi Security ,是 Spring 项目组中用来提供安全认证服务的框架。这篇文章主要介绍了SpringSecurity实现图形验证码功能,需要的朋友可以参考下
    2018-10-10
  • java代码获取数据库表里数据的总数操作

    java代码获取数据库表里数据的总数操作

    这篇文章主要介绍了java代码获取数据库表里数据的总数操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-08-08

最新评论