分析String在内存中的表现

来源:互联网 发布:stm32 串口读取数据 编辑:程序博客网 时间:2024/05/22 16:44

创建字符串的方法有两种:

         Stringstr1=”直接赋值法”

         Stringstr2=new String(“通过new关键字的方法来创建”);

那么这两种创建方式有什么区别呢?下面通过一个Demo来测试一下

package test;public class Demo {public static void main(String[] args) {String str1="java";String str2="java";String str3=new String("java");String str4=new String("java");System.out.println(str1==str2);//trueSystem.out.println(str1==str3);//falseSystem.out.println(str3==str4);//falseSystem.out.println(str3.equals(str4));//true}}

知道了它们运行后的区别了,那么他的原理是什么呢?

在执行String str1=”java”建字符串的时候,jvm会首先检查字符串常量池中是否存在该字符串对象,如果已经存在,那么就不会在创建字符串常量池中再创建了,直接返回该字符串在字符串常量池中内存地址,如果该字符串还不存在字符串常量池中,那么就会在字符串常量。池中先创建该字符串的对象,然后再返回。所以在执行String str2=”java”的时候,因为字符串常量池中已经存在”java”字符串对象了,就不会在字符串常量池中再创建了,所以在栈内存中str1和str2的内存地址都是指向”java”在字符串常量池的位置。

在执行String str3=new String(“java”)创建字符串的时候,jvm首先会检查字符串常量池中是否存在”java”的字符串,如果已经存在,则不会在字符串常量池中创建了,如果没有存在,那么就会在字符串常量池中创建”java”字符串对象,然后还会到堆内存中再创建一份字符串对象,把字符串常量池中的”java”字符串内容拷贝到内存中的字符串对象,然后返回堆内存中字符串对象的内存地址。即栈内存存储的地址是堆内存的内存地址。String str4=new String(“java”)是在堆内存中又创建了一个对象,所以str3==str4返回的是false。用一张图来表示一下。

知道这些以后就知道前三个的输出结果的原因了。那么为什么

System.out.println(str3.equals(str4));这个执行结果会返回true呢?通过看源码可以知道,在Object类中equals的方法是比较两个对象的内存地址

public boolean equals(Object obj) {return (this == obj);    }

在String类中对equals方法进行了重写,源代码是这样的。

    public boolean equals(Object anObject) {if (this == anObject) {    return true;}if (anObject instanceof String) {    String anotherString = (String)anObject;    int n = count;    if (n == anotherString.count) {char v1[] = value;char v2[] = anotherString.value;int i = offset;int j = anotherString.offset;while (n-- != 0) {    if (v1[i++] != v2[j++])return false;}return true;    }}return false;    }

在String中的equals方法中会先判断两个对象的内存地址是否相等,如果相等,那么equals就return true,如果不相等,则会进一步判断传过来的对象是不是字符串类型的,如果是则将两个字符串对象char数组,然后一个一个字符的比较,如果都相等,则返回true。


2 0
原创粉丝点击