Java中的String思考。

来源:互联网 发布:sqlplus执行sql文件 -c 编辑:程序博客网 时间:2024/06/05 13:45

今天我想对String类进行总结一下:

1>首先String类不是基本类型(char,byte,short,int ,float,double,long,boolean),该类位于java.lang包底下。该类重写了Object中的equals方法,

 public boolean equals(Object anObject) {if (this == anObject) {    return true;}if (anObject instanceof String) {    String anotherString = (String)anObject;    int n = count;    if (n == anotherString.count) {char v1[] = value;char v2[] = anotherString.value;int i = offset;int j = anotherString.offset;while (n-- != 0) {    if (v1[i++] != v2[j++])return false;}return true;    }}return false;    }

如果两个String对象的地址一致时,说明这两个对象是相同的,返回true。如果这两个String对象不是指向同一个地址时,然后比较他们中个的内容是否相同。所以当你调用equals方法时,String其实比较的是其中的内容。

2>String对象是不可改变的,看JDK时你就会发现,String类被final修饰,说明String类不可以被继承。String类中每一个看起来会修改String值的方法,实际上都是创建了一个全新的String对象,以包含修改后的字符串内容。

public static void main(String[] args) {String str1 = "a.b.c";String str2 = str1.replace(".", "\\");final int i = 1;System.out.println(str1);System.out.println(str2);}

返回结果:a.b.c
                    a\b\c

自己看见的一些面试题:

String s = "Hello";s = s + " world!";这两行代码执行后,原始的String对象中的内容到底变了没有
没有。因为String被设计成不可变(immutable)类,所以它的所有对象都是不可变对象。在这段代码中,s原先指向一个String对象,内容是 "Hello",然后我们对s进行了+操作,那么s所指向的那个对象是否发生了改变呢?答案是没有。这时,s不指向原来那个对象了,而指向了另一个 String对象,内容为"Hello world!",原来那个对象还存在于内存之中,只是s这个引用变量不再指向它了。
通过上面的说明,我们很容易导出另一个结论,如果经常对字符串进行各种各样的修改,或者说,不可预见的修改,那么使用String来代表字符串的话会引起很大的内存开销。因为 String对象建立之后不能再改变,所以对于每一个不同的字符串,都需要一个String对象来表示。这时,应该考虑使用StringBuffer类,它允许修改,而不是每个不同的字符串都要生成一个新的对象。并且,这两种类的对象转换十分容易。
同时,我们还可以知道,如果要使用内容相同的字符串,不必每次都new一个String。例如我们要在构造器中对一个名叫s的String引用变量进行初始化,把它设置为初始值,应当这样做:
public class Demo {
private String s;
...
public Demo {
s = "Initial Value";
}
...
}
而非
s = new String("Initial Value");
后者每次都会调用构造器,生成新对象,性能低下且内存开销大,并且没有意义,因为String对象不可改变,所以对于内容相同的字符串,只要一个String对象来表示就可以了。也就说,多次调用上面的构造器创建多个对象,他们的String类型属性s都指向同一个对象。
上面的结论还基于这样一个事实:对于字符串常量,如果内容相同,Java认为它们代表同一个String对象。而用关键字new调用构造器,总是会创建一个新的对象,无论内容是否相同。
至于为什么要把String类设计成不可变类,是它的用途决定的。其实不只String,很多Java标准类库中的类都是不可变的。在开发一个系统的时候,我们有时候也需要设计不可变类,来传递一组相关的值,这也是面向对象思想的体现。不可变类有一些优点,比如因为它的对象是只读的,所以多线程并发访问也不会有任何问题。当然也有一些缺点,比如每个不同的状态都要一个对象来代表,可能会造成性能上的问题。所以Java标准类库还提供了一个可变版本,即 StringBuffer。

StringStringBuffer的区别

JAVA平台提供了两个类:StringStringBuffer,它们可以储存和操作字符串,即包含多个字符的字符数据。String类表示内容不可改变的字符串。而StringBuffer类表示内容可以被修改的字符串。当你知道字符数据要改变的时候你就可以使用StringBuffer。典型地,你可以使用StringBuffer来动态构造字符数据。另外,String实现了equals方法,new String(“abc”).equals(new String(“abc”)的结果为true,StringBuffer没有实现equals方法,所以,new StringBuffer(“abc”).equals(new StringBuffer(“abc”)的结果为false

public static void main(String[] args) {String str1 = new String("abc");String str2 = new String("abc");System.out.println(str1.equals(str2));StringBuffer bf1 = new StringBuffer("abc");StringBuffer bf2 = new StringBuffer("abc");System.out.println(bf1 == bf2);}

输出结果:

true
false

StringBufferStringBuilder的区别

  StringBufferStringBuilder类都表示内容可以被修改的字符串,StringBuilder是线程不安全的,运行效率高,如果一个字符串变量是在方法里面定义,这种情况只可能有一个线程访问它,不存在不安全的因素了,则用StringBuilder。如果要在类里面定义成员变量,并且这个类的实例对象会在多线程环境下使用,那么最好用StringBuffer

如何把一段逗号分割的字符串转换成一个数组?

public static void main(String[] args) {//Split分割String str1 = new String("a,b,c");//token分割StringTokenizer token = new StringTokenizer("a,b,c",",");while(token.hasMoreTokens()){System.out.println(token.nextElement());}}

下面这条语句一共创建了多少个对象:String s="a"+"b"+"c"+"d";

public static void main(String[] args) {String s1 = "a";String s2 = s1 + "b";String s3 = "a" + "b";System.out.println(s1 == "ab");System.out.println(s3 == "ab");}

第一条语句打印的结果为false,第二条语句打印的结果为true,这说明javac编译可以对字符串常量直接相加的表达式进行优化,不必要等到运行期去进行加法运算处理,而是在编译时去掉其中的加号,直接将其编译成一个这些常量相连的结果

题目中的第一行代码被编译器在编译时优化后,相当于直接定义了一个”abcd”的字符串,所以,上面的代码应该只创建了一个String对象。写如下两行代码,

         String s = "a" + "b" + "c" + "d";

         System.out.println(s == "abcd");

最终打印的结果应该为true

 

public static void main(String[] args) {  String s1 = "abc";  String s2 = "abc";  System.out.println(s1 == s2); }

输出结果:

true

JVM创建了两个引用变量s1,s2,这两个引用变量都存放于栈中,但是只创建了一个对象,两个引用变量都指向同一个对象。

public static void main(String[] args) {String s1 = "abc";String s3 = new String("abc");System.out.println(s1 == s3);}

输出结果:

false

创建了两个引用,创建了两个对象,两个引用分别指向不同的两个引用。
以上两段代码说明,只要是用new()来新建对象的,都会在堆中创建,而且其字符串是单独存值的,即使与栈中的数据相同,也不会与栈中的数据共享。
 

原创粉丝点击