一、先比较String、StringBuffer、StringBuilder变量的HashCode值

使用System.out.println(obj.hashcode())输出的时对象的哈希码,
而非内存地址。在Java中是不可能得到对象真正的内存地址的,因为Java中堆是由JVM管理的不能直接操作。
只能说此时打印出的Hash码表示了该对象在JAVA虚拟机中的内存位置,
Java虚拟机会根据该hash码最终在真正的的堆空间中给该对象分配一个地址.
但是该地址 是不能通过java提供的api获取的

String变量连接新字符串会改变hashCode值,变量是在JVM中“连接——断开”;
StringBuffer变量连接新字符串不会改变hashCode值,因为变量的堆地址不变。
StringBuilder变量连接新字符串不会改变hashCode值,因为变量的堆地址不变。

小结:哈希码相同的变量,其内容不一定相同;内容向同变量,哈希码不一定相同;

public class Buffer_HashCode01 {
public static void main(String[] args) {
String str1="学海无涯苦作舟";
System.out.println("String 的 str1的hascode"+str1.hashCode());
str1=str1+"人间正道是沧桑"; //注意此str1的存储地址已发生变化
System.out.println("String 的 str1的hascode"+str1.hashCode());
System.out.println(str1);
StringBuffer sb1=new StringBuffer("弟子规");
System.out.println("原来 StringBuffer 的 hascode:"+sb1.hashCode());
StringBuffer sb2=sb1.append("圣人教"); //注意sb1和sb2指向同一个地址
System.out.println("添加字符串后StringBuffer 的 hascode:"+sb2.hashCode());
sb2.insert(6,"海纳百川");
StringBuilder sb=new StringBuilder("这显然是上半年");
System.out.println("改变前StringBuilder变量的hashCode值"+sb.hashCode());
sb.append("哦是的啊");
System.out.println("改变后StringBuilder变量的hashCode值"+sb.hashCode()); }
}

结果:

String 的 str1的hascode-2054942391
String 的 str1的hascode895667206
学海无涯苦作舟人间正道是沧桑
原来 StringBuffer 的 hascode:1311053135
添加字符串后StringBuffer 的 hascode:1311053135
改变前StringBuilder变量的hashCode值118352462
改变后StringBuilder变量的hashCode值118352462

二、比较String、StringBuffer、StringBuilder性能(仅在繁复操作)

String类由于Java中的共享设计,在修改变量值时使其反复改变栈中的对于堆的引用地址,所以性能低。

StringBuffer和StringBuilder类设计时改变其值,其堆内存的地址不变,避免了反复修改栈引用的地址,其性能高。

其中StringBuilder是专门类似于StringBuffer类的非线性安全类,即StringBuffer是线性安全的,适合于多线程操作;

StringBuilder是线性不安全的,适合于单线程操作,其性能比StringBuffer略高。

public class Xing_Neng_SSS01 {
public static void main(String[] args) {
long begin1 = System.currentTimeMillis();
String str = "";
for(int i=0;i<10000;i++){
str = str+i;
}
long end1 = System.currentTimeMillis();
long time1 = end1 - begin1;
System.out.println("1、String + time="+time1); long begin2 = System.currentTimeMillis();
String str2 = "";
for(int i=0;i<10000;i++){
str2 = str2.concat(i+"");
}
long end2 = System.currentTimeMillis();
long time2 = end2 - begin2;
System.out.println("2、String concat time="+time2); long begin3 = System.currentTimeMillis();
StringBuffer str3 = new StringBuffer();
for(int i=0;i<10000;i++){
str3.append(""+i);
}
long end3 = System.currentTimeMillis();
long time3 = end3 - begin3;
System.out.println("3、StringBuffer time="+time3); long begin4 = System.currentTimeMillis();
StringBuilder str4 = new StringBuilder();
for(int i=0;i<10000;i++){
str4.append(""+i);
}
long end4 = System.currentTimeMillis();
long time4 = end4 - begin4;
System.out.println("4、StringBuilder time="+time4);
}
}

结果:

1、String + time=241
2、String concat time=92
3、StringBuffer time=4
4、StringBuilder time=2

三、String共享设计

当String使用引号创建字符串时,会先去字符串池中找,找到了就返回,找不到就在字符串池中增加一个然后返回,这样由于共享提高了性能。

而new String()无论内容是否已经存在,都会开辟新的堆空间,栈中的堆内存也会改变。

下面是==来比较地址是否相等。

public class String_Equals01 {
public static void main(String[] args) {
String str1="学海无涯苦作舟人间正道是沧桑";
String str11=str1;
String str12="学海无涯苦作舟人间正道是沧桑";
String str13=new String("学海无涯苦作舟人间正道是沧桑"); System.out.println("String 的 str1 的hascode"+str1.hashCode());
System.out.println("String 的 str11的hascode"+str11.hashCode());
System.out.println("String 的 str12的hascode"+str12.hashCode());
System.out.println("String 的 str13的hascode"+str13.hashCode());
System.out.println("str1==str11 "+(str1==str11));
System.out.println("str1==str12 "+(str1==str12));
System.out.println("str1==str13 "+(str1==str13));
}
}

结果:

String 的 str1 的hascode895667206
String 的 str11的hascode895667206
String 的 str12的hascode895667206
String 的 str13的hascode895667206
str1==str11 true
str1==str12 true
str1==str13 false

Java中String、StringBuffer、StringBuilder区别与理解的更多相关文章

  1. Java中String/StringBuffer/StringBuilder区别(转)

    1.三者在执行速度方面的比较:StringBuilder >  StringBuffer  >  String 2.String <(StringBuffer,StringBuild ...

  2. java中string , StringBuffer , StringBuilder 区别

    1.String String变量的值不能改变,如果要改变String变量的值,虚拟机首先会遍历方法区中的字符串常量,如果存在需要的值,则虚拟机直接把此常量值的地址分配给String变量,如果不存在这 ...

  3. java中 String StringBuffer StringBuilder的区别

    * String类是不可变类,只要对String进行修改,都会导致新的对象生成. * StringBuffer和StringBuilder都是可变类,任何对字符串的改变都不会产生新的对象. 在实际使用 ...

  4. java中String,StringBuffer,StringBuilder之间的区别

    文章转载自:http://www.cnblogs.com/frankliiu-java/archive/2010/07/05/1771537.html String是固定长度的字符串,如果要发生变化必 ...

  5. 探秘Java中String、StringBuilder以及StringBuffer

    探秘Java中String.StringBuilder以及StringBuffer 相信String这个类是Java中使用得最频繁的类之一,并且又是各大公司面试喜欢问 到的地方,今天就来和大家一起学习 ...

  6. 转:String StringBuffer StringBuilder区别

    转自:http://www.iteye.com/topic/522167 作者:每次上网冲杯Java时,都能看到关于String无休无止的争论.还是觉得有必要让这个讨厌又很可爱的String美眉,赤裸 ...

  7. JAVA中String和StringBuilder类的特点及使用

    转自:https://www.imooc.com/code/2202 仅做个人学习记录之用,侵删! 什么是 Java 中的字符串 在 Java 中,字符串被作为 String 类型的对象处理. Str ...

  8. String&StringBuffer&StringBuilder区别

    String  String类是final类故不可以继承,也就意味着String引用的字符串内容是不能被修改.String有两种实例化方式:    (1)直接赋值(例中,String str = &q ...

  9. java 中String与StringBuilder 效率

    之前印象中string与stringbuilder操作时,如果多次改变string就使用stringbuilder,效率会提高: 今天实际遇到了问题,亲身经历过之后,这性能不是一般的影响啊:不是同一个 ...

  10. java中String、StringBuilder、StringBuffer三者的区别

    在Java项目开发中,字符串是最长使用的数据类型,而有关字符串的String.StringBuilder.StringBuffer三者又常常让人分不清楚什么时候该使用哪个. 特此整理一下. Strin ...

随机推荐

  1. Spring Autowired 注入失败总是Null

    报错:NullPointerException 分析:错误原因是注入失败? <context:annotation-config/> <context:component-scan ...

  2. 规约模式Specification的学习

    最近一直在看DDD开发  规约似乎用得很普遍. 但是还是理解不了.所以记录下学习的进度.- 规约(Specification)模式 目的:查询语句和查询条件的分离 写了一个关于规约的模拟小程序 cla ...

  3. js简单放羊式单元测试-上

    这是看了很多js单元测试资料后第一次自己做单元测试,因为资料都在介绍工具怎么使用,js单元测试的工具是在是太多了,各种风格,各种支持的,新的旧的,so 还是自己动手来体验一次 简单 是我给自己的需求很 ...

  4. MySQL sql_safe_updates 分析

    我在练习MySQL操作语句时,使用一条完全没有错误的语句: update students set name='drake' where name='chuan'; 却报了如下错误: Error Co ...

  5. python 获取网页图片

    re为正则表达式模组 re.findall在字符串中查找所有匹配的模式,返回一个list urllib2提供了使用简单的url工具 urllib2.urlopen发送url请求,返回一个文件流 imp ...

  6. 20145205 《Java程序设计》第8周学习总结

    教材学习内容总结 第十五章 通用API 15.1 日志 日志API简介 java.util.logging包提供了日志功能相关类与接口,不必额外配置日志组件,就可在标准Java平台使用是其好处.使用日 ...

  7. 【Java】实战Java虚拟机之五“开启JIT编译”

    今天开始实战Java虚拟机之五“开启JIT编译” 总计有5个系列 实战Java虚拟机之一“堆溢出处理” 实战Java虚拟机之二“虚拟机的工作模式” 实战Java虚拟机之三“G1的新生代GC” 实战Ja ...

  8. 青蛙跳100级台阶算法,完整可运行,php版本

    /* 算法题目 * 2016年4月11日16:11:08 * 一只青蛙,一次可以跳1步,或者2步,或者3步,现在要跳100级台阶,请问青蛙有多少种上100级台阶的跳法 * 1步的有$n 2步的有$m ...

  9. 快速理解几种常用的RAID磁盘阵列级别

    我发现周围不少人在学习和理解RAID磁盘阵列的原理时,找了很多专业的资料来看,但是因为动手的机会比较少,因此看完以后还是似懂非懂,真正遇到实际的方案设计的时候,还是拿不定主意. 因此,我结合自己在过去 ...

  10. NC57,NC63-NC二开经验总结

    版主2010级市场营销专业本科生 2013年8月入达内培训Java相关技术 12月入职,做用友NC的二次开发工作 2015年4月离职,4中下旬入职一家互联网金融企业 下面是做NC二开期间积累的一些常用 ...