最近有几个Java字符串的问题一直没有搞明白,记录下来,以备以后查看。

一、 字符串做为参数会不会改变其值

public void aMethod() {
  String str = "original";
      modifyStr(str) ;
      System.out.println(str);
}
 
public void modiftStr(String str) {
  str = str + "has been changed";
}

此时输出为什么?

开始以为String为引用类型,所以在传递的时候传递的是地址,因此在函数modifyStr中改变了str的值。但是经过测试实际情况并不是这样,输出的仍然是original,可见str的值并没有因为modifyStr而改变。这是为什么呢? 其实本质的原因在于String是一个封装类。

封装类,或者叫做包装类。大家都知道java中有8中基本数据类型,int, float等等,但是基本数据类型并不完全支持面向对象的编程,因此在JDK中针对各种基本类型定义了各自的引用类型,称之为封装类。如Integer 是int的封装类,Float是float的封装类。封装类的操作实际上都是对其对应的基本类型进行操作实现的。 大家看一下String对象的构造函数:

public String(String original) { 

                int size = original.count; 

                char[] originalValue = original.value; 

                char[] v; 

                if (originalValue.length > size) { 

                        // The array representing the String is bigger than the new 

                        // String itself.    Perhaps this constructor is being called 

                        // in order to trim the baggage, so make a copy of the array. 

                            int off = original.offset; 

                            v = Arrays.copyOfRange(originalValue, off, off+size); 

                } else { 

                        // The array representing the String is the same 

                        // size as the String, so no point in making a copy. 

                        v = originalValue; 

                } 

                this.offset = 0; 

                this.count = size; 

                this.value = v; 

}

可见String起始就是char[]的封装类型。这就能解释为何modifyStr后str的值为什么没有变化了。

 

二、 new 一个String对象和给一个String 对象赋值有何区别

String a = "original";

String b = new String("original");

a创建的对象位于java中的常量池中,而b对象位于内存堆中。

 

三、 string str = “a” + "b" + "c"一共创建了几个对象?

一个。因为JVM做了优化,“a”,"b","c"直接是常量池中的常量了,并没有再创建“a”,"b","c"等对象。

 

四、 String Stringbuilder Stringbuffer的区别

 

String是固定长度的,不可变的。所以每次对String进行编辑时(如str = str + "abc")实际是有创建了一个新的对象,比较耗时。而stringbuffer则是变长的,可以很好的执行对字符串的改变等操作,所以效率上要优于string. StringBUilder是从5.0以后有的类,和stringBuffer的区别是stringbuffer是线程安全的,而stringbuilder是线程不安全的。如在单线程中使用的话使用stringbuilder更有效率,因为他不必保证线程的安全性。一般来讲,三者的效率关系为:String<StringBuffer<StringBuilder

但是在个别情况下可能string的效率更高:

String S1 = “This is only a” + “ simple” + “ test”;
 StringBuffer Sb = new StringBuilder(“This is only a”).append(“ simple”).append(“ test”);

这种情况下会发现S1要比Sb的效率高很多。为什么呢?起始这是JVM玩的一个把戏,KVM直接将 “This is only a” + “ simple” + “ test”看成是“This is only a simple test”,当然效率就更高了。