前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >新手学JAVA(三)----StringBuilder类

新手学JAVA(三)----StringBuilder类

作者头像
令仔很忙
发布2018-09-14 16:11:06
4010
发布2018-09-14 16:11:06
举报
文章被收录于专栏:令仔很忙令仔很忙

上一篇文章新手学JAVA(二)----String类与StringBuffer类的区别中了解到,String的值是不可变的,这就导致

每次对String的操作都会生成新的String对象,不仅效率低下,而且大量浪费有限的内存空间,StringBuffer是可变

类,和线程安全的字符串操作类,任何对它指向的字符串的操作都不会产生新的对象。

  StringBuffer类和StringBuilder类功能基本相似。算是两个双胞胎。

  下面主要说两点

  第一点  线程安全

  StringBuffer  线程安全

  StringBuilder 线程不安全

  关于线程安全的知识,正在学习,刚接触,没有太深入的了解,在这知识稍微的提一下。

  线程安全——如果你的代码所在的进程中有多个线程在同时运行,而这些线程可能会同时运行这段代码。如果每次

运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,就是线程安全的。

  StringBuffer类和StringBuilder类两者没有很大的区别,但是在线程安全方面,StringBuffer允许多线程进行字

符操作。这是因为在源代码中StringBuffer的很多方法都被关键字synchronized(这个关键字是为线程同步机制设定

的。) 修饰了,而StringBuilder没有。

  简单的说一说synchronized的含义:

  每一个类对象都对应一把锁,当某个线程A调用类对象O中的synchronized方法M时,必须获得对象O的锁才能够执行

M方法,否则线程A阻塞。一旦线程A开始执行M方法,将独占对象O的锁。使得其它需要调用O对象的M方法的线程阻

塞。只有线程A执行完毕,释放锁后。那些阻塞线程才有机会重新调用M方法。这就是解决线程同步问题的锁机制。

  因此,多线程编程中StringBuffer比StringBuilder要安全的多。

  有一点需要注意的是,有的人会问,String类是不是也不安全? 事实上不存在这个问题,String是不可变的。线

程对于堆中指定的一个String对象只能读取,无法修改。还有什么不安全的?

  第二点  效率问题

  一般情况下,速度从快到慢:StringBuilder>StringBuffer>String,这种比较是相对的,不是绝对的。

  举个简单的例子:

代码语言:javascript
复制
public class TestCharacter{
	final static int time=100;  //循环次数
	
	public TestCharacter(){
		
		}
	
	public void test(String s){
		long begin = System.currentTimeMillis();
		for(int i=0;i<time;i++){
			s+="add";
			}
		long over=System.currentTimeMillis();
		System.out.println("操作"+s.getClass().getName() +"类型使用的时间为:" +(over-begin)+"毫秒");
		}
		
	public void test(StringBuffer s){
		long begin = System.currentTimeMillis(); 
		for(int i=0; i<time; i++){ 
			s.append("add"); 
		} 
		long over = System.currentTimeMillis(); 
		System.out.println("操作"+s.getClass().getCanonicalName()+"类型使用的时间为:"+(over-begin)+"毫秒"); 

		}
		
	public void test(StringBuilder s){ 
		long begin = System.currentTimeMillis(); 
		for(int i=0; i<time; i++){ 
			s.append("add"); 
		} 
		long over = System.currentTimeMillis(); 
		System.out.println("操作"+s.getClass().getName()+"类型使用的时间为:"+(over-begin)+"毫秒"); 
		} 

	/*对 String 直接进行字符串拼接的测试*/ 
	public void test2(){ 
		String s2 = "abcd"; 
		long begin = System.currentTimeMillis(); 
		for(int i=0; i<time; i++){ 
			String s = s2 + s2 +s2; 
		} 
		long over = System.currentTimeMillis(); 
		System.out.println("操作字符串对象引用相加类型使用的时间为:"+(over-begin)+"毫秒"); 
	}
	 
	public void test3(){ 
		long begin = System.currentTimeMillis(); 
		for(int i=0; i<time; i++){ 
			String s ="abcd" + "abcd" + "abcd"; 
		} 
		long over = System.currentTimeMillis(); 
		System.out.println("操作字符串相加使用的时间为:"+(over-begin)+"毫秒"); 
	} 
	
	public static void main(String[] args){ 
		String s1 = "abcd"; 
		StringBuffer st1 = new StringBuffer("abcd"); 
		StringBuilder st2 = new StringBuilder("abcd"); 
		TestCharacter tc = new TestCharacter(); 
		tc.test(s1); 
		tc.test(st1); 
		tc.test(st2); 
		tc.test2(); 
		tc.test3(); 
	} 

}

  下面是循环50000次,10000次,1000次,100次的运行结果:

  总结 

  (1).如果要操作少量的数据用 = String

  (2).单线程操作字符串缓冲区 下操作大量数据 = StringBuilder

  (3).多线程操作字符串缓冲区 下操作大量数据 = StringBuffer 

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2015年03月30日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  •   第一点  线程安全
  •   第二点  效率问题
  •   总结 
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档