1,请运行以下示例代码StringPool.java,查看其输出结果。如何解释这样的输出结果?从中你能总结出什么?
在Java中,内容相同的字串常量(“Hello”)只保存一份以节约内存,所以s0,s1,s2实际上引用的是同一个对象。
编译器在编译s2一句时,会去掉“+”号,直接把两个字串连接起来得一个字串(“Hello”)。这种优化工作由Java编译器自动完成。
当直接使用new关键字创建字符串对象时,虽然值一致(都是“Hello”),但仍然是两个独立的对象。
给字串变量赋值意味着:两个变量(s1,s2)现在引用同一个字符串对象“a”!
String对象的内容是只读的,使用“+”修改s1变量的值,实际上是得到了一个新的字符串对象,其内容为“ab”,它与原先s1所引用的对象”a”无关,所以,s1==s2返回false;
代码中的“ab”字符串是一个常量,它所引用的字符串与s1所引用的“ab”对象无关。
String.equals()方法可以比较两个字符串的内容。
若在第四行加一行代码如下:System.out.println(s1=”a”);则其运行结果为true。
2,请查看String.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类的equals()方法虽然接收的参数是Object类型,但是比较的时候却只和同样为String类型的对象比较,如果不是String类型,直接返回false。之所以参数为Object,而不是String,只是为了重写Object类的equals()方法的需要。实现过程:(1)String类中的equals首先比较地址,如果是同一个对象的引用,可知对象相等,返回true。(2)若果不是同一个对象,equals方法挨个比较两个字符串对象内的字符,只有完全相等才返回true,否则返回false。比较方法为:先计算两个String类型对象的长度判断是否相等,相等继续往下走否则返回false,若为相等用两个char型数组接受String对象的每一个字符,offset为数组第一个字符的索引,在这里初始值为0,然后用循环判断两个数组是否相等。其中n--!=0可认为是int x=n;x!=0;x--。
3,参看示例 StringMisc.java,其中展示了以下String类的重要方法与字段
Length():获取字串长度,无参调用后返回长度。
charAt():获取指定位置的字符,类似于数组参数为0到字符串长度-1.
getChars():获取从指定位置起的子串复制到字符数组中(它有四个参数,在示例中有介绍)1.被拷贝字符在字串中的起始位置 2.被拷贝的最后一个字符在字串中的下标再加1 3.目标字符数组 4.拷贝的字符放在字符数组中的起始下标
replace():子串替换参数与两个分别为被替换的子串和替换的子串。
toUpperCase()、 toLowerCase():大小写转换,直接调用无参
trim():去除头尾空格,直接调用无参:
toCharArray():将字符串对象转换为字符数组,直接调用无参。
4,请阅读JDK中String类上述方法的源码,模仿其编程方式,编写一个MyCounter类,它的方法也支持上述的“级联”调用特性,其调用示例为:
MyCounter counter1=new MyCounter(1);
MyCounter counter2=counter1.increase(100).decrease(2).increase(3);
….
public class LeiSI { int data; public LeiSI(int data)//构造方法 { this.data=data; } public LeiSI()//空构造方法 { } //想要实现“级联”调用,要在方法里创建对象,把返回值设成对象,从而返回的对象还可以再调用 public LeiSI increase(int d)//实现data加的方法,并返回对象 { LeiSI a=new LeiSI(); a.data=data+d; return a; } public LeiSI decrease(int d)// 实现data减的方法,并返回对象 { LeiSI a=new LeiSI(); a.data=data-d; return a; } public static void main(String[] args) { LeiSI counter1=new LeiSI(1);//创建对象counter1 LeiSI counter2=counter1.increase(100).decrease(2).increase(3);//创建对象counter2 System.out.println(counter2.data);//输出counter对象的data结果 } }