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

来源:互联网 发布:犀牛地方门户源码官网 编辑:程序博客网 时间:2024/06/09 21:48

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中,但是两个不同的对象,地址值不同
}


0 0
原创粉丝点击