Java学习笔记之String、StringBuffer和StringBuilder
来源:互联网 发布:免费联系人恢复软件 编辑:程序博客网 时间:2024/05/16 19:55
更多博文可参考我的个人博客:贱贱的梦想
基本概念
String
:此类代表字符串常量,它们的值在创建之后不能更改。StringBuffer
:是一个线程安全的可变字符序列,它与String
一样,在内存中保存的都是一个有序的字符串序列(char
类型的数组),不同点是StringBuffer
对象的值是可变的。StringBuilder
:与StringBuffer
类基本相同,都是可变字符串系列,不同点是StringBuilder
是线程不安全的。
分析
- 简单的说,
String
类型和StringBuilder
、StringBuffer
类型的主要性能区别在于String
是不可变的对象。这是由于,每次对String
类型进行改变的时候,其实都等同于生成了一个新的String
对象,然后将指针指向新的String
对象。所以经常改变内容的字符串最好不要用String
,因为每次生成对象都会对系统性能产生影响,特别当内存中无引用对象多了以后, JVM 的 GC 就会开始工作,那速度是一定会相当慢的。 - 对
StringBuffer
类型进行改变时,就是对其对象本身进行改变,而不是生成新的对象,再改变对象引用。在一般情况下,推荐使用StringBuffer
,特别是字符串对象经常改变的情况下。 StringBuffer
在程序中可将字符串缓冲区安全地用于多线程,而且在必要时可以对这些方法进行同步。而StringBuilder
的使用场景是单线程,不保证同步,该类被设计用作StringBuffer
的一个简易替换,用在字符串缓冲区被单个线程使用时。在这种情况下,它比StringBuffer
要快。
equal 和 ==
==
用于在比较两个对象的时候,检查两个引用是否指向了同一块内存。 equals()
是object
的方法,默认情况下,它与==
一样,比较的地址。但是当equal
被重载之后,根据设计,equal
会比较对象的value
。而这个是java
希望有的功能。String
类就重写了这个方法。
例1:
String obj1 = new String("hello");String obj2 = new String("hello");if(obj1 == obj2) System.out.println(true);else System.out.println(false);
输出为false
。
例2:
String obj1 = new String("hello");String obj2 = new String("hello");if(obj1.equals(obj2)) System.out.println(true);else System.out.println(false);
输出为true
。
例3:obj1
与obj2
指向同一块内存
String obj1 = new String("hello");String obj2 = obj1;if(obj1 == obj2) System.out.println(true);else System.out.println(false);
输出为true
。
例4:一种特殊情况
String obj1 = "hello";String obj2 = "hello";if(obj1 == obj2) System.out.println(true);else System.out.println(false);
输出为true
。这是因为程序在运行的时候会创建一个字符串缓冲池。当使用 String obj1 = "xyz";
这样的表达是创建字符串的时候(非new
这种方式),程序首先会在这个 String
缓冲池中寻找相同值的对象,在 String str1 = "xyz";
中,obj1
先被放到了池中,所以在 obj2
被创建的时候,程序找到了具有相同值的 str1
并将 s2
引用 s1
所引用的对象 "xyz"
。
例5:
String obj1 = new String("hello");String obj2 = "hello";if(obj1 == obj2) System.out.println(true);else System.out.println(false);
输出为false
。由输出结果可知,obj1
对象不在字符串缓冲池中,故在创建obj2
时,在缓冲池中没有找到相同值的对象,因此obj1
与obj2
所指向的引用是不同的。
在Java中,对象都创建在堆内存中。
例6:字符串操作符+
public class Test { public static void main(String[] args) { int a = 1; int b = 2; System.out.println( a + b + ""); System.out.println("" + a + b ); System.out.println( a + b + "" + a + b); System.out.println("" + ( a + b )); }} /** output3123123*/
第一个输出语句:先进行整数求和运算,再将运算结果转换成String输出;
第二个输出语句:由于第一个变量类型为String,后面的变量都得转换成String,然后进行String组合,而不是先进行整数求和运算;
第三个输出语句:由于第一个变量类型不是String,所以先进行求和运算,然后遇到了String类型,运算结果要进行转换,并且后面的int也需要进行转换;
第四个输出语句:虽然第一个变量类型为String,但是后面的括号控制了表达式的赋值顺序,以使得int类型的变量在显示之前确实进行了求和操作。
CharSequence源码分析(基于jdk1.7.79)
package java.lang;public interface CharSequence { /** * 返回字符序列的长度 */ int length(); /** * 返回指定索引的char值 */ char charAt(int index); /** * 返回该序列的子序列 * @param start 开始位置的索引值(包括) * @param end 结束位置的索引值(不包括) * 说明:当start=end时,返回一个空序列 */ CharSequence subSequence(int start, int end); /** * 返回一个包含该序列中字符的字符串,该字符串与序列的顺序相同 */ public String toString();}
StringBuilder源码分析(基于jdk1.7.79)
package java.lang;public final class StringBuilder extends AbstractStringBuilder implements java.io.Serializable, CharSequence{ /** use serialVersionUID for interoperability */ static final long serialVersionUID = 4383685877147921099L; /** * 构造一个不带任何字符的字符串生成器,其初始容量为16个字符 */ public StringBuilder() { super(16); } /** * 构造一个不带任何字符的字符串生成器,其初始容量由capacity参数设定 * 说明:其capacity参数值不能小于0,否则会抛出异常NegativeArraySizeException */ public StringBuilder(int capacity) { super(capacity); } /** * 构造一个字符串生成器,并初始化指定的字符串内容。其初始容量为指定字符串的 * 长度加上16 * 说明:其str不能为null,否则会抛出异常NullPointerException */ public StringBuilder(String str) { super(str.length() + 16); append(str); } /** * 构造一个字符串生成器,其包含与指定的字符序列相同的字符。其初始容量为字符 * 序列的长度加上16 * 说明:其seq不能为null,否则会抛出异常NullPointerException */ public StringBuilder(CharSequence seq) { this(seq.length() + 16); append(seq); } /** * 追加Object参数的字符串表示形式到此序列 * 说明:如果obj为null,则追加"null";如果obj不为null,则追加obj.toString() */ public StringBuilder append(Object obj) { return append(String.valueOf(obj)); } /** * 将指定的字符串追加到此序列 * 说明:如果str为null,则追加"null" */ public StringBuilder append(String str) { super.append(str); return this; } /** 将指定的StringBuilder参数追加到此序列 */ private StringBuilder append(StringBuilder sb) { if (sb == null) // 如果sb为null,则追加"null" return append("null"); int len = sb.length(); int newcount = count + len; // count为之前的字符长度 if (newcount > value.length) // 若新的字符长度newcount大于存储容量 expandCapacity(newcount); // 则进行扩容 sb.getChars(0, len, value, count); // 将sb复制到原字符序列的尾部 count = newcount; // 更新字符长度 return this; } /** 将指定的StringBuffer追加到此序列 */ public StringBuilder append(StringBuffer sb) { super.append(sb); return this; } /** 将指定的CharSequence追加到此序列 */ public StringBuilder append(CharSequence s) { if (s == null) s = "null"; if (s instanceof String) return this.append((String)s); if (s instanceof StringBuffer) return this.append((StringBuffer)s); if (s instanceof StringBuilder) return this.append((StringBuilder)s); return this.append(s, 0, s.length()); } /** 将指定的CharSequence子序列追加到此序列 */ public StringBuilder append(CharSequence s, int start, int end) { super.append(s, start, end); return this; } /** 将char数组参数的字符串表示形式追加到此序列 */ public StringBuilder append(char[] str) { super.append(str); return this; } /** 将char数组参数的子数组的字符串表示形式追加到此序列 */ public StringBuilder append(char[] str, int offset, int len) { super.append(str, offset, len); return this; } /** * 将boolean参数的字符串表示形式追加到此序列 * 说明:若b为true,则追加"true";若b为false,则追加"false" */ public StringBuilder append(boolean b) { super.append(b); return this; } /** 将char字符的字符串表示形式追加到此序列 */ public StringBuilder append(char c) { super.append(c); return this; } /** 将int参数的字符串表示形式追加到此序列 */ public StringBuilder append(int i) { super.append(i); return this; } /** 将int参数的字符串表示形式追加到此序列 */ public StringBuilder append(long lng) { super.append(lng); return this; } /** 将float参数的字符串表示形式追加到此序列 */ public StringBuilder append(float f) { super.append(f); return this; } /** 将double参数的字符串表示形式追加到此序列 */ public StringBuilder append(double d) { super.append(d); return this; } /** 将codePoint参数的字符串表示形式追加到此序列 */ public StringBuilder appendCodePoint(int codePoint) { super.appendCodePoint(codePoint); return this; } /** * 移除此序列的子字符串中的字符 * 说明:包含起始索引但不包括结束索引 */ public StringBuilder delete(int start, int end) { super.delete(start, end); return this; } /** 移除此序列指定位置上的字符 */ public StringBuilder deleteCharAt(int index) { super.deleteCharAt(index); return this; } /** * 使用给定的String中的字符替换此序列的子字符串中的字符 * 该子字符串从指定的start处开始,一直到索引end-1处的字符 */ public StringBuilder replace(int start, int end, String str) { super.replace(start, end, str); return this; } /** * 将str数组参数的子数组的字符串表示形式插入到此序列中 * 子数组从指定的offset(包括offset)开始,包括len个char * 子数组的字符将被插入到index所指示的位置 */ public StringBuilder insert(int index, char[] str, int offset, int len) { super.insert(index, str, offset, len); return this; } /** 将Object参数的字符串表示形式插入到此字符序列中 */ public StringBuilder insert(int offset, Object obj) { return insert(offset, String.valueOf(obj)); } /** 将字符串插入到此序列 */ public StringBuilder insert(int offset, String str) { super.insert(offset, str); return this; } /** 将char数组参数的字符串表示形式插入此序列中 */ public StringBuilder insert(int offset, char[] str) { super.insert(offset, str); return this; } /** 将指定的CharSequence插入此序列中 */ public StringBuilder insert(int dstOffset, CharSequence s) { if (s == null) s = "null"; if (s instanceof String) return this.insert(dstOffset, (String)s); return this.insert(dstOffset, s, 0, s.length()); } /** 将指定 CharSequence 的子序列插入此序列中 */ public StringBuilder insert(int dstOffset, CharSequence s, int start, int end) { super.insert(dstOffset, s, start, end); return this; } /** 将 boolean 参数的字符串表示形式插入此序列中 */ public StringBuilder insert(int offset, boolean b) { super.insert(offset, b); return this; } /** 将 char 参数的字符串表示形式插入此序列中 */ public StringBuilder insert(int offset, char c) { super.insert(offset, c); return this; } /** 将 int 参数的字符串表示形式插入此序列中 */ public StringBuilder insert(int offset, int i) { return insert(offset, String.valueOf(i)); } /** 将 long 参数的字符串表示形式插入此序列中 */ public StringBuilder insert(int offset, long l) { return insert(offset, String.valueOf(l)); } /** 将 float 参数的字符串表示形式插入此序列中 */ public StringBuilder insert(int offset, float f) { return insert(offset, String.valueOf(f)); } /** 将 double 参数的字符串表示形式插入此序列中 */ public StringBuilder insert(int offset, double d) { return insert(offset, String.valueOf(d)); } /** 返回指定子字符串在此字符串中第一次出现处的索引 */ public int indexOf(String str) { return indexOf(str, 0); } /** 返回指定子字符串在此字符串中第一次出现处的索引,从指定的索引开始*/ public int indexOf(String str, int fromIndex) { return String.indexOf(value, 0, count, str.toCharArray(), 0, str.length(), fromIndex); } /** 返回最右边出现的指定子字符串在此字符串中的索引 */ public int lastIndexOf(String str) { return lastIndexOf(str, count); } /** 从指定位置开始,返回最右边出现的指定子字符串在此字符串中的索引 */ public int lastIndexOf(String str, int fromIndex) { return String.lastIndexOf(value, 0, count, str.toCharArray(), 0, str.length(), fromIndex); } /** 将此字符序列用其反转形式取代 */ public StringBuilder reverse() { super.reverse(); return this; } /** 返回此序列中数据的字符串表示形式 */ public String toString() { // Create a copy, don't share the array return new String(value, 0, count); } /** * Save the state of the <tt>StringBuilder</tt> instance to a stream * (that is, serialize it). * * @serialData the number of characters currently stored in the string * builder (<tt>int</tt>), followed by the characters in the * string builder (<tt>char[]</tt>). The length of the * <tt>char</tt> array may be greater than the number of * characters currently stored in the string builder, in which * case extra characters are ignored. */ private void writeObject(java.io.ObjectOutputStream s) throws java.io.IOException { s.defaultWriteObject(); s.writeInt(count); s.writeObject(value); } /** * readObject is called to restore the state of the StringBuffer from * a stream. */ private void readObject(java.io.ObjectInputStream s) throws java.io.IOException, ClassNotFoundException { s.defaultReadObject(); count = s.readInt(); value = (char[]) s.readObject(); }}
- Java学习笔记之String、StringBuffer和StringBuilder
- java学习笔记之String,StringBuilder和StringBuffer
- 【Java学习笔记】String、StringBuffer和StringBuilder
- java基础学习笔记之String、StringBuffer、StringBuilder
- java学习笔记-String,StringBuffer,StringBuilder浅析
- java基础笔记之String、StringBuffer、StringBuilder
- 【JAVA_SE学习笔记】String、StringBuffer和StringBuilder
- 【Java学习】String、StringBuffer和StringBuilder
- JAVA基础笔记之StringBuilder和StringBuffer
- java之String、StringBuffer、StringBuilder
- java之String,StringBuffer,StringBuilder,
- Java之String、StringBuffer、StringBuilder
- Java之String、StringBuffer、StringBuilder
- java String,StringBuffer和StringBuilder
- java String,StringBuffer和StringBuilder
- Java String、StringBuffer和StringBuilder
- [Java]String、StringBuilder和StringBuffer
- Java学习之StringBuffer和StringBuilder
- linux目录和文件名的命名规则
- 解决fastjson内存对象重复/循环引用json错误
- mysql 优化(3)聚簇索引和非聚簇索引
- 【Leetcode】Verify Preorder Serialization of a Binary Tree
- 资料收藏
- Java学习笔记之String、StringBuffer和StringBuilder
- 操作系统常见面试题总结
- 【Leetcode】Binary Tree Zigzag Level Order Traversal
- java设计模式1 单例模式
- 【Leetcode】 Simplify Path
- 子类对象实例化全过程
- 使用AVFoundation扫描二维码
- MySQL 5.7.10 自动备份、自动清理旧备份集
- Sflow介绍