一、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 3308 最长连续上升区间

    LCIS Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  2. hdu5601 BestCoder Round #67 (div.2)

    N*M bulbs  Accepts: 94  Submissions: 717  Time Limit: 10000/5000 MS (Java/Others)  Memory Limit: 655 ...

  3. 服务器使用nginx做代理,通过HttpServletRequest获取请求用户真实IP地址

    首先,在nginx配置中添加如下配置 server { listen ; server_name www.wenki.info; #要访问的域名 charset utf8; location / { ...

  4. java的迭代器详解

    迭代器的引出 在jdk1.5版本之前是没有 foreach的,然而1.5版本就加上了foreach,而引入的新的foreach功能并不是在jvm上进行改进的因为代价太高,甲骨文工程师想到了一个比较好的 ...

  5. VMWare 学习目录

    Linux介绍 Linux入门--个人感想 Google怎么用linux 初入Linux Windows XP硬盘安装Ubuntu 12.04双系统图文详解 实例讲解虚拟机3种网络模式(桥接.nat. ...

  6. 谷歌Chrome浏览器之No Sandbox

     想着还是要把这个分享出来,前两天,早上过来,Chrome打开后,输入网址回车,毫无反应,不加载,不跳转,打不开浏览器设置.总之就是除了能打开Chrome浏览器,不能进行任何其他操作,关闭重开也是这样 ...

  7. mybatis逆向工程,转载别人的,很清楚

    转载博客地址:http://www.cnblogs.com/selene/p/4650863.html

  8. spring的事务配置方法

    spring事务的配置有两种方式 1.xml配置的声明式事务配置 (1)配置数据源信息dataSource(使用阿里的数据源) <bean id="dataSource" c ...

  9. os和sys模块的区别及其常用方法总结

    官方解释:os: This module provides a portable way of using operating system dependent functionality. 翻译:提 ...

  10. PHP 实例 AJAX 与 MySQL

    AJAX 数据库实例 下面的实例将演示网页如何通过 AJAX 从数据库读取信息: 实例   Person info will be listed here... 实例解释 - MySQL 数据库 在上 ...