java源码分析(2)-String(1)

时间:2023-02-26 00:14:13

String源码分析:

1.String类不可被继承

public final class String implements java.io.Serializable, Comparable<String>, CharSequence{}
由String的类源码可知,String由final修饰,故String不能被继承

2.String的值不可更改

 private final char value[];
public String() {
this.value = new char[0];
}
String的value属性被final修饰,只能在构造器执行时被赋值,将不能再更改,因此所有对String字符串的修改(如追加字符串,删除部分字符串,截取字符串)都不是在原来的对象基础上修改,而是新建一个String对象修改并返回,这会造成原对象被废弃,浪费资源且性能较差(特别是追加字符串和删除部分字符串),若遇到字符串将被频繁修改的情况,建议不要使用String,改用StringBuffer或StringBuilder。

3.String构造器

public String() {
this.value = new char[0];无参构造器,默认value为一个长度为0的数组,所以为null;
}
public String(String original) {
this.value = original.value;
this.hash = original.hash;
}
public String(char value[]) {
this.value = Arrays.copyOf(value, value.length);
}
public String(char value[], int offset, int count) {
if (offset < 0) {
throw new StringIndexOutOfBoundsException(offset);//起始值下标小于0,抛异常
}
if (count < 0) {
throw new StringIndexOutOfBoundsException(count);//取值长度小于0,抛异常
}
if (offset > value.length - count) {
throw new StringIndexOutOfBoundsException(offset + count);//起始值下标加长度大于数组长度,抛异常
}
this.value = Arrays.copyOfRange(value, offset, offset+count);
}
public String(int[] codePoints, int offset, int count) {
if (offset < 0) {
throw new StringIndexOutOfBoundsException(offset);
}
if (count < 0) {
throw new StringIndexOutOfBoundsException(count);
}
if (offset > codePoints.length - count) {
throw new StringIndexOutOfBoundsException(offset + count);
}
//精确计算String所需要的长度
final int end = offset + count;
int n = count;
for (int i = offset; i < end; i++) {
int c = codePoints[i];
if (Character.isBmpCodePoint(c))
continue;
else if (Character.isValidCodePoint(c))
n++;
else throw new IllegalArgumentException(Integer.toString(c));
}

//提取每一位的字符,并将其放入String字符串
final char[] v = new char[n];

for (int i = offset, j = 0; i < end; i++, j++) {
int c = codePoints[i];
if (Character.isBmpCodePoint(c))
v[j] = (char)c;
else
Character.toSurrogates(c, v, j++);
}
<pre name="code" class="java"><span style="font-family: Arial, Helvetica, sans-serif;"> this.value = v;</span>
}

4.charAt(int i)

用于取得字符串上下标为i的字符

5.equals()方法

public boolean equals(Object anObject) {//该方法重写equals()方法,不再是比较地址值,而是比较字符串的值是否相同
if (this == anObject) {
return true;//若比较的两个对象引用地址值相同,则为同一个对象,值当然相同
}
if (anObject instanceof String) {//String与另一个对象能比较的前提是,对方也属于String类型
String anotherString = (String) anObject;
int n = value.length;
if (n == anotherString.value.length) {//直接先比较长度,若不相等,则一定不等,效率很高
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {//将两个String对象的值放入数组中,遍历比较,全部相同才表示相同
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}

6.String Pool

String Pool为java内置的区域,当String不是以new String(),形势创建时,该字符串便会被存入String Pool,这个字符串会变为公共资源,当另一个引用需要指向与该字符串相同的字符串时,java不会再创建新的字符串,而是直接指向同一个字符串,请看下面的代码:
public static void main(String[] args) {
String a="hello";
String b="hello";
String c=new String ("hello");
String d=new String ("hello");
System.out.println(a==b);//true,hello被创建在String Pool中,a和b会指向同一个对象,
System.out.println(a==c);//a指向的hello被创建在String Pool中,而c所指向的对象被创建在heap中,两者为不同的对象,地址值不同
System.out.println(d==c);//c和d所指的对象都被创建在heap中,但是两个不同的对象,地址值不同
}