java字符串池(string pool)和字符串堆(heap)内存分配

(转自https://blog.csdn.net/bigLiu66/article/details/88783913

java运行环境有一个字符串池(string pool),由String类维护。


执行语句 String str = "abc" 时,首先查看字符串池中是否存在字符串"abc" ,如果存在则直接将"abc"地址赋给str ,如果不存在则先在字符串池中新建一个字符串"abc",然后再将其赋给str。


执行语句 String str = new String("abc") 时,不管字符串池中是否存在"abc" ,直接新建一个字符串"abc"(注意:新建的字符串"abc" 不是在字符串池中),然后将其赋给str。


前一语句的效率高,后一语句的效率低,因为新建字符串占用内存空间。String str = new String();创建了一个空字符串,与String str = new String("");相同。


public String intern()

返回字符串对象的规范化表示形式。一个初始为空的字符串池,它由类String私有地维护。当调用intern()方法时,如果字符串池中已经包含一个等于此String 对象的字符串 (用equals(Object)方法确定),则返回字符串池中的字符串。否则,将此String 对象添加到字符串池中,并返回此String 对象的引用。它遵循以下规则:对于任意两个字符串s 和 t,当且仅当 s.equals(t)为true时,s.intern() == t.intern() 才为true。


String.intern();

再补充一点:存在于.class 文件中的常量池,在运行期间被jvm(java virtual machine)装载,并且可以扩充。String 的 intern()方法就是扩充常量池的一个方法;当一个String 实例(instance)str调用intern()方法时,java 查找常量池中是否有相同 unicode 的字符串常量,如果有,则返回其引用,如果没有,则在常量池中增加一个unicode 等于str 的字符串并返回其引用。


简单例子:

String s = "kvill";

String s1 = new String("kvill");

String s2 = new String("kvill");

System.out.println(s == s1);

s1.intern();

s2 = s2.intern();

System.out.println(s == s1);

System.out.println(s == s1.intern());

System.out.println(s == s2);

输出结果为:

False

False   //虽然执行了s1.intern(),但是它的返回值并没有赋给s1

True

True


最后再破除一个错误的理解:

有人说,"使用String.intern()方法可以将一个String类保存到一个全局的String表中,如果具有相同值的unicode 字符串已经在这个表中,那么该方法返回表中已有字符串的地址,如果在表中没有相同unicode的字符串,则将自己的地址注册到表中",如果我们把这个全局的String表理解成常量池的话,最后一句话"如果在表中没有相同值的字符串,则将自己的地址注册到表中" 是错的。

简单例子:

String s1 = new String("kvill");

String s2 = s1.intern();

System.out.println(s1 == s1.intern());

System.out.println(s1 + " " + s2);

System.out.println(s2 == s1.intern());

输出结果是:

False

kvill kvill

True

我们没有声明一个"kvill"常量,所以常量池中一开始没有"kvill"的,当我们调用s1.intern()后,就在常量池中新添加了一个"kvill"常量,原来的不在常量池中的"kvill"仍然存在,也就不是"把自己的地址注册到常量池中"了。



例子1):

String str1 = "java"; // str1指向字符串池

String str2 = "blog"; // str2指向字符串池


String s = str1 + str2; // s是指向堆中值为"javablog"的对象,+ 运算符会在堆中建立起来两个String对象,这两个对象分别是"java","blog",也就是说从字符串池中复制这两个值,然后在堆中创建两个对象,然后再建立对象s,然后将"javablog"的堆地址赋给s。 // 这条语句总共创建了多少个对象?


System.out.println(s == "javablog"); // 结果为False

jvm(java virtual machine)确实对形如String str1 = "java"; 的String对象放在常量池中,但是它是在编译时那么做的,而String s = str1 + str2; 是在运行时才知道的,也就是说str1 + str2 是在堆里创建的,所以结果为false了。


如果改成以下两种方式:

String s = "java" + "blog";  // 直接将"javablog"放入字符串池中

System.out.println(s == "javablog");  // 结果为true

String s = str1 + "blog"; // 不放入字符串池,而是在堆中分配

System.out.println(s == "javablog"); // 结果为false




引用变量与对象的区别:

字符串"abc" 是一个String 对象,字符串池(pool of literal strings)和堆(heap)中存放着字符串对象


一、引用变量与对象

A a;

这个语句声明了一个类A的引用变量a,而对象一般通过new关键字创建,所以a仅仅是一个引用变量,而不是对象


二、java中所有的字符串文字(字符串常量)都是一个String类的对象。有人(特别是c 程序员)在一些场合喜欢把字符串"当做/看成"字符数组,这也没有办法,因为字符串与字符数组存在一些内在的联系。事实上,它与字符数组是两种完全不同的对象。


三、字符串对象的创建

由于字符串对象的大量使用(它是一个对象,一般而言对象总是在堆(heap)中分配内存),java中为了节省内存空间和运行时间(如比较字符串时,== 比 equals()方法快),在编译阶段就把所有的字符串文字放到一个字符串池(pool of literal strings)中,而运行时字符串池成为常量池的一部分。字符串池的好处,就是该池中所有相同的字符串常量被合并,只占用一个空间。


我们知道,对两个引用变量,使用 == 判断它们的值(引用)是否相等,即是否指向同一个对象:

String s1 = "abc";

String s2 = "abc";

if(s1 == s2) System.out.println("s1,s2 refer to the same object");

else System.out.println("trouble");

这里的输出显示,两个字符串文字保存为一个对象,就是说,上面的代码只在字符串池(string pool)中创建了一个对象。


现在看看String s = new String("abc");语句,这里"abc"本身就是pool中的一个对象,而在运行时执行new String()时,将pool中的对象复制一份放到heap中,并且把heap中的这个对象的引用交给s 持有。ok,这条语句就创建了2个String对象。


String s1 = new String("abc");

String s2 = new String("abc");

if(s1 == s2) { } // 不会执行的语句

这里用 == 判断就可知,虽然两个对象的"内容"相同(equals()判断),但两个引用变量所持有的引用不同,上面的代码创建了几个String Object? (三个,pool中一个,heap中两个)




综上所述:

创建字符串有两种方式:两种内存区域(pool vs heap)

1. "" 引号创建的字符串在字符串池中

2. new, new关键字创建字符串时首先查看字符串池(string pool)中是否有相同值的字符串,如果有,则拷贝一份到堆(heap)中,然后返回堆中的地址;如果字符串池中没有,则在堆中创建一份,然后返回堆(heap)中的地址(注意,此时不需要从堆中复制到池中,否则,将使得堆中的字符串永远是池中的子集,导致浪费字符串池的内存空间)

3. 对字符串进行赋值时,如果右操作数含有一个或一个以上的字符串引用时,则在堆中再建立一个字符串对象,返回引用;如String s = str1 + "blog";

比较两个已经存在于字符串池中字符串对象可以用 == 进行,拥有比equals操作更快的速度。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • 从网上复制的,看别人的比较全面,自己搬过来,方便以后查找。原链接:https://www.cnblogs.com/...
    lxtyp阅读 1,345评论 0 9
  • 本文是我自己在秋招复习时的读书笔记,整理的知识点,也是为了防止忘记,尊重劳动成果,转载注明出处哦!如果你也喜欢,那...
    波波波先森阅读 829评论 1 6
  • 概述 在分析字符串常量池之前,先来分析一下java的内存区域,然后再各种的情况分析一下各种情况下的情况; 在《深入...
    riverhh阅读 827评论 0 2
  • 前言 RTFSC (Read the fucking source code )才是生活中最重要的。我们天天就是要...
    二毛_coder阅读 455评论 1 1
  • 这是一个有点惨烈的故事,有一个悲剧的结局,比起大圆满,我更喜欢这个结局。 司徒玦最终也没有趴在姚起云的床头,哭着喊...
    fy12阅读 1,948评论 0 2