标签:
当String变量需要经常变换其值时,应该考虑使用StringBuffer类,以提高程序效率?
假设String s=new String ("wo");String s1=new String("de");
s=s+s1;
System.out.println(s);结果为wode?
首先在栈中有个"s"变量指向堆中的"wo"对象...
栈中"s1"变量指向堆中的"de"对象
当执行到s = s + s1;
系统重新在堆中new一个更大的数组出来,然后将"wo"和"de"都复制进去,然后栈中的"s"指向这个新new出来的数组...
所谓的不可变是指:它没有在原数组“wo”上进行修改,而是新建了个更大数组进行扩展,
也就是说,这时候堆里还是有“wo”这个对象数组存在的,只不过这个时候"s"变量不在指向"wo"这个数组了,
而是指向了新new出来的数组,这就是和StringBuffered的区别,后者是在原数组上进行修改,改变了原数组的值,
StringBuffered不是通过新new一个数组去复制,而是在原数组基础上进行扩展...再让变量指向原数组....
String是我们经常用到的一个类型,其实有时候觉得写程序就是在反复的操作字符串,这是C的特点,
在java中,jdk很好的封装了关于字符串的操 作。
今天主要讲的是三个类String 、StringBuffer 、 StringBuilder .
这三个类基本上满足了我们在不同情景下使用字符串的需求。
先说,第一个String。
JDK的解释是 “Strings are constant; their values cannot be changed after they are created”
也就是说String对象一旦被创建就是固定不变的了(你一定有问题,但请先等一等,耐心读下去),
这样的一点好处就是可以多线程之间访 问,因为只读不写。
一般情况下我们以下面两种方式创建一个String对象
两种方式是有区别的,这和java的内存管理有关,前面已经说过,string创建之后是不可变的,
所以按照第一种方式创建的字符串会放在栈里,更确切的是常量池中,常量池就是用来保存在编译阶段确定好了大小的数据,
一般我们定义的int等基本数据类型就保存在这里。
其具体的一个流程就是,编译器首先检查常量池,看看有没有一个“string”,如果没有则创建。如果有的话,
则则直接把str1指向那个位置。
第二种创建字符串的方法是通过new关键字,还是java的内存分配,java会将new的对象放在堆中,这一部分
对象是在运行时创建的对象。所以我们每一次new的时候,都会创建不同的对象,即便是堆中已经有了一个一模
一样的。
写一个小例子
1 String str1 = "string"; 2 String str4 = "string"; 3 String str2 = new String("string"); 4 String str3 = new String("string"); 5 6 /*用于测试两种创建字符串方式的区别*/ 7 System.out.println(str1 == str4); 8 System.out.println(str2 == str3); 9 System.out.println(str3 == str1); 10 11 str3 = str3.intern(); //一个不常见的方法 12 System.out.println(str3 == str1);
这个的运行结果是
true //解释:两个字符串的内容完全相同,因而指向常量池中的同一个区域
false //解释:每一次new都会创建一个新的对象
false // 解释: 注意==比较的是地址,不仅仅是内容
true //介绍一下intern方法,这个方法会返回一个字符串在常量池中的一个地址,如果常量池中有与str3内容相
同的string则返回那个地址,如果没有,则在常量池中创建一个string后再返回。实际上,str3现在指向了str1
的地址。
这就是让人纠结的string了,现在你可以说话了。。。很多人有这样的疑问就是既然string是不变的,那么
为什么str1 + "some"是合法的,其实,每次对string进行修改,都会创建一个新的对象。
所以如果需要对一个字符串不断的修改的话,效率是非常的低的,因为堆的好处是可以动态的增加空间,劣
势就是分配新的空间消耗是很大的,比如我们看下面的测试。
1 long start = System.currentTimeMillis(); 2 3 for(int i = 0; i < 50000; i++) 4 { 5 str1+= " "; 6 } 7 8 long end = System.currentTimeMillis(); 9 System.out.println("the run time is "+(end -start)+" ms");
我的机器上运行结果是the run time is 3538 ms 如果你把循环的次数后面再增加几个0就会更慢。因为每
一次循环都在创建心的对象,那么JDK如何解决这个问题?
下面就要说第二个类StringBuffer。
StringBuffer是一个线程安全的,就是多线程访问的可靠保证,最重要的是他是可变的,也就是说我们要操
作一个经常变化的字符串,可以使用 这个类,基本的方法就是append(与string的concat方法对应)和insert
方法,至于怎么使用,就不多讲了,大家可以自己查看API。
1 StringBuilder sb = new StringBuilder("string builder"); 2 StringBuffer sf = new StringBuffer("string buffer"); 3 4 long start = System.currentTimeMillis(); 5 6 for(int i = 0; i < 50000; i++) 7 { 8 //str1+= " "; 9 sb.append(" "); 10 } 11 12 long end = System.currentTimeMillis(); 13 System.out.println("the run time is "+(end -start)+" ms");
测试一下,这次只需要8ms,这就是效率。
那么接下来,就要问StringBuilder是干什么的,其实这个才是我们尝使用的,这个就是在jdk 1.5版本后面
添加的新的类,前面说StringBuffer是线程同步的,那么很多情况下,我们只是使用一个线程,那个同步势必带
来一个效率的问 题,StringBuilder就是StringBuffer的非线程同步的版本,二者的方法差不多,只是一个线程
安全(适用于多线程)一个没有线程安全 (适用于单线程)。
其实看了一下jdk源代码就会发现,StringBuffer就是在各个方法上加上了关键字syncronized
以上就是对三个字符串类的一个总结,总之不要在这上面纠结。。。。。。不想介绍太多的方法,总觉得那
样会把一篇博客弄成API文档一样,而且还非常的繁琐。都是些体会,希望有所帮助。起码不要再纠结,尤其是
面试。。。。
面试题4:替换空格
它原先的想法是从头开始遍历,遇到空格就替换,这样的话后面的字符就会移动,有些字符会移动多次,算法效率较低,时间复杂度为O(n^2)
所以提出一种新的想法,就是从后面开始遍历,先算出替换之后的长度是多少,然后用一个指针A指向那个地方,另一个指针B指向原来字符数组的末尾,然后一个一个复制过去,当B遇到空格的时候,在A那里添加“%20”。
大概思路就是这样子。
但是问题就出在这里,字符数组的长度是不可变的,至少在java中是这样,所以我们要新建一个数组吗?
但是新建一个数组的话,我们只要从头开始一个一个遍历,然后遇到空格就在新数组中加入“%20”,这样时间复杂度不也是O(n)吗
那这道题不就没有意义了?
所以我想问的是 大家觉得剑指offer的题都适合用java做吗?
错误的想法如下:
正确的想法,但是StringBuffer是如何工作的?
C语言代码
1 // ReplaceBlank.cpp : Defines the entry point for the console application. 2 // 3 4 // 《剑指Offer——名企面试官精讲典型编程题》代码 5 // 著作权所有者:何海涛 6 7 #include "stdafx.h" 8 #include <string> 9 10 /*length 为字符数组string的总容量*/ 11 void ReplaceBlank(char string[], int length) 12 { 13 if(string == NULL && length <= 0) 14 return; 15 16 /*originalLength 为字符串string的实际长度*/ 17 int originalLength = 0; 18 int numberOfBlank = 0; 19 int i = 0; 20 while(string[i] != ‘\0‘) 21 { 22 ++ originalLength; 23 24 if(string[i] == ‘ ‘) 25 ++ numberOfBlank; 26 27 ++ i; 28 } 29 30 /*newLength 为把空格替换成‘%20‘之后的长度*/ 31 int newLength = originalLength + numberOfBlank * 2; 32 if(newLength > length) 33 return; 34 35 int indexOfOriginal = originalLength; 36 int indexOfNew = newLength; 37 while(indexOfOriginal >= 0 && indexOfNew > indexOfOriginal) 38 { 39 if(string[indexOfOriginal] == ‘ ‘) 40 { 41 string[indexOfNew --] = ‘0‘; 42 string[indexOfNew --] = ‘2‘; 43 string[indexOfNew --] = ‘%‘; 44 } 45 else 46 { 47 string[indexOfNew --] = string[indexOfOriginal]; 48 } 49 50 -- indexOfOriginal; 51 } 52 } 53 54 void Test(char* testName, char string[], int length, char expected[]) 55 { 56 if(testName != NULL) 57 printf("%s begins: ", testName); 58 59 ReplaceBlank(string, length); 60 61 if(expected == NULL && string == NULL) 62 printf("passed.\n"); 63 else if(expected == NULL && string != NULL) 64 printf("failed.\n"); 65 else if(strcmp(string, expected) == 0) 66 printf("passed.\n"); 67 else 68 printf("failed.\n"); 69 } 70 71 // 空格在句子中间 72 void Test1() 73 { 74 const int length = 100; 75 76 char string[length] = "hello world"; 77 Test("Test1", string, length, "hello%20world"); 78 } 79 80 // 空格在句子开头 81 void Test2() 82 { 83 const int length = 100; 84 85 char string[length] = " helloworld"; 86 Test("Test2", string, length, "%20helloworld"); 87 } 88 89 // 空格在句子末尾 90 void Test3() 91 { 92 const int length = 100; 93 94 char string[length] = "helloworld "; 95 Test("Test3", string, length, "helloworld%20"); 96 } 97 98 // 连续有两个空格 99 void Test4() 100 { 101 const int length = 100; 102 103 char string[length] = "hello world"; 104 Test("Test4", string, length, "hello%20%20world"); 105 } 106 107 // 传入NULL 108 void Test5() 109 { 110 Test("Test5", NULL, 0, NULL); 111 } 112 113 // 传入内容为空的字符串 114 void Test6() 115 { 116 const int length = 100; 117 118 char string[length] = ""; 119 Test("Test6", string, length, ""); 120 } 121 122 //传入内容为一个空格的字符串 123 void Test7() 124 { 125 const int length = 100; 126 127 char string[length] = " "; 128 Test("Test7", string, length, "%20"); 129 } 130 131 // 传入的字符串没有空格 132 void Test8() 133 { 134 const int length = 100; 135 136 char string[length] = "helloworld"; 137 Test("Test8", string, length, "helloworld"); 138 } 139 140 // 传入的字符串全是空格 141 void Test9() 142 { 143 const int length = 100; 144 145 char string[length] = " "; 146 Test("Test9", string, length, "%20%20%20"); 147 } 148 149 int _tmain(int argc, _TCHAR* argv[]) 150 { 151 Test1(); 152 Test2(); 153 Test3(); 154 Test4(); 155 Test5(); 156 Test6(); 157 Test7(); 158 Test8(); 159 Test9(); 160 161 return 0; 162 }
转载:
http://developer.51cto.com/art/201204/327109.htm
http://bbs.csdn.net/topics/390869449
标签:
原文地址:http://www.cnblogs.com/Mokaffe/p/4315901.html