一、String的JVM内存分配测试与分析 
        
        String a="a";
        String b="b";
        String c="ab";
        String d="ab";
        String e=a+b;
        String f = "a" + "b";
        final String p = "a";
        final String q = "b";
        String m = p + q;
        String n = p + "b";
        String str1 = new String("ab");
        String str2 = new String("ab");

        System.out.println("c==d?" + (c==d));
        System.out.println("d==e?" + (d==e));
        System.out.println("c==f?" + (c==f));
        System.out.println("c==m?" + (c==m));
        System.out.println("c==n?" + (c==n));
        System.out.println("d==str1?" + (d==str1));
        System.out.println("str1==str2?" + (str1==str2));
        System.out.println("e==str1?" + (e==str1));

输出结果:

c==d?true
d==e?false
c==f?true
c==m?true
c==n?true
d==str1?false
str1==str2?false
e==str1?false

程序中用来存放数据的内存分为四块

1、全局区(静态区)(static)

2、文字常量区 :常量字符串就是放在这块区域,即是我们常说起的常量池。

3、栈区(stack):存放函数的参数值,局部变量的值等。

4、堆区(heap) : 存放对象



当我们定义字符串

String a = "a";

a在栈区,“a”是字符串常量,在常量池中

String b = "b";

b在栈区,“b”在常量池

String c="ab";

c在栈区,“ab”在常量池

String d="ab";

d在栈区,这个时候常量池里已经有"ab",所以直接使用已经有的那个“ab”

所以这个时候c和d都指向的常量池里面的同一个“ab”

String e=a+b;

e在栈区,a+b实际上产生了一个新的String对象,既然是String对象,所以结果“ab”放在堆区中,即e指向的是堆里的“ab”

这样的情况下,c==d为true,c==e为false

////////////////////////////////////

另外,如果定义的是字符串对象

String str1 = new String("ab");

str1在栈区,创建的“ab”字符串对象在堆区

String str2 = new String("ab");

str2在栈区,又创建的一个新的“ab”对象也在堆区,不过和刚才的“ab”不是同一个。

相当于堆区中有两个字符串对象,不过正好内容都是“ab”而已。

所以str1==str2为false



常量池里面放着的常量字符串可以重复使用,但是必须是你直接使用的该字符串,像a+b这种形式虽然得到的结果是“ab”,但并不是使用的字符串常量“ab”。

关于String e=a+b,还有一种说法是经过编译器优化之后,结果为“ab”,则编译器直接从常量池中拿出“ab”常量,将引用赋值给e。我使用的JDK是1.6.0_34,得出的结果是d==e?false,说明这个版本的JVM没有进行上述的编译器优化。
(后期修正:对于String e = a + b;编译器是存在优化的。但是编译器的优化是在编译期,由于a和b都是变量字符串,在编译期无法确定其具体值,故编译期无法优化。如果a和b都使用了final修饰并且赋了初始值,即a和b是有初始值的常量字符串,则在编译期就能确定其值,编译器就会将e的值优化为"ab"。关于JVM常量池和字符串优化更详细的解释,请参考我的另一篇博文:String放入运行时常量池的时机与String.intern()方法解惑.

以上绝大部分内容摘抄了http://blog.sina.com.cn/s/blog_4b622a8e0100c296.html处的分析。



二、StringBuilder&StringBuffer

看了以下两个帖子,都写得挺详细的,大家可以去看看:

http://blog.csdn.net/kingzone_2008/article/details/9220691

http://www.cnblogs.com/dolphin0520/p/3778589.html


看了看源码,自己的一点理解:

1)
StringBuffer是线程安全的,StringBuilder是非安全的,在大部分情况下我们都不需要考虑多线程问题,所以用StringBuilder效率会高一点点,但当需要多线程的时候,就要使用StringBuffer了。以下英文是StringBuilder源码里的注释,也给出了使用建议。

This class provides an API compatible

 * with <code>StringBuffer</code>, but with no guarantee of synchronization.

 * This class is designed for use as a drop-in replacement for

 * <code>StringBuffer</code> in places where the string buffer was being

 * used by a single thread (as is generally the case).   Where possible,

 * it is recommended that this class be used in preference to

 * <code>StringBuffer</code> as it will be faster under most implementations.
2)
        StringBuilder sb  = new StringBuilder();
        sb.append( "AAAA");
        String appendStr = null;
        sb.append(appendStr);
        sb.append( "BBBB");
        System. out.println( "sb:" +
sb.toString());

         控制台输出结果:sb:AAAAnullBBBB
         这是因为对于append方法,当参数为null时,StringBuilder&StringBuffer中的处理是默认将“null”字符串加入sb的后面。 


String&StringBuilder&StringBuffer总结的更多相关文章

  1. 深入源码剖析String,StringBuilder,StringBuffer

    [String,StringBuffer,StringBulider] 深入源码剖析String,StringBuilder,StringBuffer [作者:高瑞林] [博客地址]http://ww ...

  2. String, StringBuilder, StringBuffer问题

    1. 区别 String为字符串常量,而StringBuilder和StringBuffer都是字符串变量,其中StringBuilder线程非安全,StringBuffer线程安全. 每次对 Str ...

  3. String StringBuilder StringBuffer区别

    String StringBuilder StringBuffer String类是final类,不可以被继承,且它的成员方法也是final方法,当一个字符串对象进行操作操作时,任何的改变不会影响到这 ...

  4. difference among String,StringBuilder,StringBuffer

    difference among String,StringBuilder,StringBuffer String常用构造函数 String(byte[] bytes) String(byte[] b ...

  5. JDK源码分析系列---String,StringBuilder,StringBuffer

    JDK源码分析系列---String,StringBuilder,StringBuffer 1.String public final class String implements java.io. ...

  6. java中String StringBuilder StringBuffer比较和效率(性能)测试

    string stringbuilder stringbuffer三者的区别 从JDK源码看,String.StringBuilder.StringBuffer都是存放在char[] 数组字符串. 简 ...

  7. string,stringbuilder,stringbuffer用法

    总结:1.如果要操作少量的数据用 = String   ==================================>字符串常量2.单线程操作字符串缓冲区 下操作大量数据 = Strin ...

  8. java中string stringbuilder stringbuffer 的区别

    1. String 类 String的值是不可变的,这就导致每次对String的操作都会生成新的String对象,不仅效率低下,而且大量浪费有限的内存空间. String a = "a&qu ...

  9. String,StringBuilder,StringBuffer

    (转:http://blog.csdn.net/rmn190/article/details/1492013)   String 字符串常量StringBuffer 字符串变量(线程安全)String ...

  10. String Stringbuilder Stringbuffer的区别

    String 字符串常量StringBuffer 字符串变量(线程安全)StringBuilder 字符串变量(非线程安全) 简要的说, String 类型和 StringBuffer 类型的主要性能 ...

随机推荐

  1. hdu 1402 FFT(模板)

    A * B Problem Plus Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Other ...

  2. 一则利用内核漏洞获取root权限的案例【转】

    转自:https://blog.csdn.net/u014089131/article/details/73933649 目录(?)[-] 漏洞描述 漏洞的影响范围 漏洞曝光时间 漏洞产生的原因 漏洞 ...

  3. wpf中静态资源和动态资源的区别

    静态资源(StaticResource)指的是在程序载入内存时对资源的一次性使用,之后就不再访问这个资源了. 动态资源(DynamicResource)指的是在程序运行过程中然会去访问资源.

  4. 函数的形参和实参之arguments对象

    当函数调用函数时候传入的实参比函数声明时候制定的形参要少时候,剩余的形参就设置成了undefined.例如 function getPropertyNames(o,/*optional*/a){ va ...

  5. struts框架从.jsp页面直接访问action

    <%@ page language="java" pageEncoding="UTF-8"%><%String path = request. ...

  6. jvm(四):垃圾回收

    垃圾回收我们主要从以下三个方面进行描述 垃圾对象的判断 目前判断对象为垃圾对象有两种方法:引用计数法,可达性分析法,目前普遍是的是可达性分析法 可达性分析法的实现原理: 定义gcroot一直往下找,如 ...

  7. 41. First Missing Positive(困难, 用到 counting sort 方法)

    Given an unsorted integer array, find the first missing positive integer. For example, Given [1,2,0] ...

  8. API说明书规范

    目录 1       前言 1.1         编写目的 1.2        预期读者 1.3         关于API设计开发 2       API公共说明 3       文档API索引 ...

  9. 一个任务:(小甲鱼python视频第29讲) 代码整理与总结

    任务:将文件(record.txt)中的数据进行分割,并安装以下规则保存起来.  1.小甲鱼的对话单独保存为boy_*.txt的文件(去掉"小甲鱼:")  2.小客服的对话单独保存 ...

  10. 高效update方案

    --方案1:如果有索引,先把索引删除后,再update,最后把索引重新创建一下因为索引对update影响很大. --方案2:1.create table newA as select id,name, ...