JVM-String比较-字节码分析(转载)

public class a {
    public static void main(String[] args)throws Exception{

    }
    public static void aa(){
        String s1="a";//内存在方法区的常量池
        String s2="b";//内存在方法区的常量池
        String s12 = "ab";//内存在方法区的常量池
        String s3 = s1 + s2;//s3的内存所在???
        p(s3==s12);//false
    }
    public static void bb(){
        String s1="a"+"b";//s1的内存所在???
        String s2 = "ab";//内存在方法区的常量池
        p(s1==s2);//true
    }public static void p(Object obj){
        System.out.println(obj);
    }
}

这是我们经常碰到的烦人的String比较问题,要得到答案,就要弄清楚aa()方法中的s3的内存在哪里?,和bb()方法中的s1的内存在哪里?

不多说,贴上a.class文件反编译的字节码指令:

首先是 aa()方法:

public static void aa();
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=4, args_size=0                                                  共4个本地变量空间
         0: ldc           #3                  // String a                            将字符串"a"从常量池中推送至栈顶
         2: astore_0                                                                  将栈顶引用类型(即字符串"a")存入第一个本地变量
         3: ldc           #4                  // String b                              将字符串"b"从常量池推送至栈顶
         5: astore_1                                                                  将栈顶引用类型(即字符串"b")存入第二个本地变量
         6: ldc           #5                  // String ab                             将字符串"ab"从常量池推送至栈顶
         8: astore_2                                                                  将栈顶引用类型(即字符串"ab")存入第三个本地变量
         9: new           #6                  // class java/lang/StringBuilder         创建StringBuilder对象sb,并将引用值压入栈顶
        12: dup                                                                       复制栈顶数值,并将复制值压入栈顶
        13: invokespecial #7                  // Method java/lang/StringBuilder.       调用对象的初始化方法
"<init>":()V
        16: aload_0                                                                   将第一个本地变量(即字符串"a")推送至栈顶
        17: invokevirtual #8                  // Method java/lang/StringBuilder.       调用实例方法sb.append("a");
append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        20: aload_1                                                                   将第二个本地变量(即字符串"b")推送至栈顶
        21: invokevirtual #8                  // Method java/lang/StringBuilder.       调用实例方法sb.append("b");
append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
        24: invokevirtual #9                  // Method java/lang/StringBuilder.       调用实例方法sb.toString(),并将结果【Java堆地址】放在栈顶
toString:()Ljava/lang/String;
        27: astore_3                                                                   将栈顶引用类型(即堆地址)存入第四个本地变量
        28: aload_3                                                                    将第四个本地变量(即堆地址)推送至栈顶
        29: aload_2                                                                    将第三个本地变量(即字符串"ab")推送至栈顶
        30: if_acmpne     37                                                           比较栈顶两引用数值,结果不同跳转(当然不同啦)
        33: iconst_1
        34: goto          38
        37: iconst_0                                                                   将int类型 0 压入栈顶
        38: invokestatic  #10                 // Method java/lang/Boolean.valueO       调用静态方法Boolean.valueOf();实现基本数据类型->包装类型自动转换
f:(Z)Ljava/lang/Boolean;
        41: invokestatic  #11                 // Method p:(Ljava/lang/Object;)V        调用静态方法p(false);//输出false
        44: return                                                                     从当前方法返回void

针对其中的一些解释:(下面的数字是字节码偏移量)

24       为何在sb.toString()我说的是【堆地址】,大家看源码就知道了。

//这是StringBuilder的源码,返回的是堆上的字符串地址
public String toString() {
    return new String(value, 0, count);
}

所以在aa()方法中,s3的内存其实在Java堆上,s12在方法区的常量池上,所以两者不相等。

37      boolean到底分配几个字节,在这里大家可以看到。

如果为true,编译器翻译的字节码是iconst_1,意思将int类型1存入栈顶,所以单个引用boolean值时,分配4个字节,和int相同。(数组boolean没测试,不清楚)

如果为false,编译器翻译的字节码是iconst_0,意思将int类型0存入栈顶。

38      在这里我们还能看到自动类型转换的身影,这里是基本数据类型boolean->包装类Boolean的自动类型转换,实际调用的就是Boolean.valueOf()静态方法,这是因为下面的p()方法里面需要的是Object引用类型,所以进行了自动类型转换。

然后是 bb()方法:

public static void bb();
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=2, args_size=0                                                   两个本地变量空间
         0: ldc           #5                  // String ab                             将字符串"ab"从常量池中推送至栈顶
         2: astore_0                                                                   将栈顶引用类型(字符串"ab")存入第一个本地变量
         3: ldc           #5                  // String ab                             将字符串"ab"从常量池中推送至栈顶
         5: astore_1                                                                   将栈顶引用类型(字符串"ab")存入第一个本地变量
         6: aload_0                                                                    将第一个本地变量("ab")推送至栈顶
         7: aload_1                                                                    将第二个本地变量("ab")推送至栈顶
         8: if_acmpne     15                                                           比较栈顶两引用类型数值,结果不同跳转(这里当然相同啦)
        11: iconst_1                                                                   将int类型 1 推送至栈顶
        12: goto          16                                                           无条件跳转到16字节码偏移量
        15: iconst_0
        16: invokestatic  #10                 // Method java/lang/Boolean.valueO       调用静态方法Boolean.valueOf();并将返回的Boolean类型的true压入栈顶
f:(Z)Ljava/lang/Boolean;
        19: invokestatic  #11                 // Method p:(Ljava/lang/Object;)V        调用静态方法p(true);输出true
        22: return                                                                     从当前方法返回void

时间: 2024-07-28 16:04:56

JVM-String比较-字节码分析(转载)的相关文章

通过字节码分析JDK8中Lambda表达式编译及执行机制

关于Lambda字节码相关的文章,很早之前就想写了,[蜂潮运动]APP 产品的后端技术,能快速迭代,除了得益于整体微服架构之外,语言层面上,也是通过Java8的lambda表达式的运用以及rxJava响应式编程框架,使代码更加简洁易维护,调用方式更加便捷.本文将介绍JVM中的方法调用相关的字节码指令,重点解析JDK7(JSR-292)之后新增的invokedynamic指令给lambda表达式的动态调用特性提供的实现机制,最后再探讨一下lambda性能方面的话题. 方法调用的字节码指令 在介绍i

Java并发编程原理与实战八:产生线程安全性问题原因(javap字节码分析)

前面我们说到多线程带来的风险,其中一个很重要的就是安全性,因为其重要性因此,放到本章来进行讲解,那么线程安全性问题产生的原因,我们这节将从底层字节码来进行分析. 一.问题引出 先看一段代码 package com.roocon.thread.t3; public class Sequence { private int value; public int getNext(){ return value++; } public static void main(String[] args) { S

通过字节码分析java中的switch语句

在一次做题中遇到了switch的问题,由于对switch执行顺序的不了解,在这里简单的通过字节码的方式理解一下switch执行顺序(题目如下): public class Ag{ static public int i=10; public static void main(String []args){ switch(i){ default: System.out.println("this is default"); case 1: System.out.println("

JVM-String比较-字节码分析

一道String字符串比较问题引发的字节码分析 public class a { public static void main(String[] args)throws Exception{ } public static void aa(){ String s1="a";//内存在方法区的常量池 String s2="b";//内存在方法区的常量池 String s12 = "ab";//内存在方法区的常量池 String s3 = s1 +

There is no getter for property named &#39;*&#39; in &#39;class java.lang.String&#39;之源码分析

There is no getter for property named '*' in 'class java.lang.String',此错误之所以出现,是因为mybatis在对parameterType="String"的sql语句做了限制,假如你使用<when test="username != null">这样的条件判断时,就会出现该错误,不过今天我们来刨根问底一下. 一.错误再现 想要追本溯源,就需要错误再现,那么假设我们有这样一个sql查询

JVM总括三-字节码、字节码指令、JIT编译执行

JVM总括三-字节码.字节码指令.JIT编译执行 java文件编译后的class文件,java跨平台的中间层,JVM通过对字节码的解释执行(执行模式,还有JIT编译执行,下面讲解),屏蔽对操作系统的依赖.一个字节(8位)可以储存256中不同的指令,这样的指令就是字节码,java所有指令有200个左右,这些指令组成了字节码文件(.class). 一.字节码的主要指令: .class文件里面的十六进制文件(如:图一),其中CAFE BABE是标志这个文件为java的编译后的文件,00000034代表

字节码分析与操作

1.1什么是字节码 https://zh.wikipedia.org/wiki/Java%E5%AD%97%E8%8A%82%E7%A0%81 Java所宣称的一次编译处处运行就是靠的字节码技术,java文件编译后会生成字节码文件.class,供jvm使用.字节码文件是由十六进制值组成,两个十六进制为一组,以一个字节为单位进行读取. 编译 javac *.java 反编译javap -c -verbose *.class 1.2.字节码结构 public class ByteCodeDemo {

透过字节码分析java基本类型数组的内存分配方式。

我们知道java中new方式创建的对象都是在堆中创建的,而局部变量对应的值存放在栈上.那么java中的int [] arr={1,2,3}是存放在什么地方的呢,int []arr = new int[3]又是存放在什么地方的呢, 下面我们通过编写两个小例子,通过查看生成的字节码文件,来了解jvm会如何来处理这两种情况的. 1.int[] arr = new int[3]示例 public class ArrayTest { public static void main(String[] arg

java中i=i++字节码分析

原文出处: Ticmy 1 2 int i = 0; i = i++; 结果还是0为什么? 程序的执行顺序是这样的:因为++在后面,所以先使用i,"使用"的含义就是i++这个表达式的值是0,但是并没有做赋值操作,它在整个语句的最后才做赋值,也就是说在做了++操作后再赋值的,所以最终结果还是0 让我们看的更清晰点: 1 2 int i = 0;//这个没什么说的 i = i++;//等效于下面的语句: 1 2 3 int temp = i;//这个temp就是i++这个表达式的值 i++