JAVA String、StringBuilder、和StringBuffer的区别,及如何使用
String类
一、String类的理解和创建对象
- String对象用于保存字符串的,也就是一组字符序列
- 字符常量对象是用双引号括起来的字符序列。
- 字符串的字符使用Unicode字符编码,一个字符(不区分字符还是汉字)占两个字节
- String类常用构造器
String s2 = new String(String original);
String s3 = new String(char[] a);
String s4 = new String(char[] a,int startIndex,int count)
String s5 = new String(byte[] b)
-
String 实现了 接口Serializable 【String可以串型化,可以在网络传输】
- 接口Comparable 【String可以相互比较大小】
-
String 是一个finally类,不能被其他类继承
-
String 有属性 private final byte[] value; 用于存放字符串内容
-
即:String本质是一个value数组,且这个数组的地址不能被改变
但是它里面存的字符可以改变
-
-
这个value是一个final类型的,不可修改的(指的是一地址不可修改)
- 即value不能指向新的地址,但是字符的内容可以变化
final char[] value = {'a','b','c'}; char[] v2 = {'t','o','m'}; // value =v2;//不可以修改value的地址 // Cannot assign a value to final variable 'value' value[0] = 'H';
String关系图:
二、String类创建的方式
1)方式一:直接赋值String s = "ywl"
2)方式二:调用构造器String s = new String("ywl");
两种创建String对象的区别
- 方式一:先从常量池查看是否有
"ywl"
数据空间,如果有,直接指向;如果没有则重新创建,然后指向。S最终指向的是常量池的空间地址 - 方式二:先在堆中创建空间,里面有了value属性,指向常量池的
ywl
空间,如果没有则在常量次中创建'ywl'
,并让value指向这个地址,如果有,让value属性指向现有的字符串的地址.
画出两种方式的内存分布图
测试题
题目1:
知识点:
- 当调用intern方法时,如果池已经包含与
equals(Object)
方法确定的相当于此String
对象的字符串,则返回来自池的字符串。否则,此String
对象将添加到池中,并返回对此String
对象的引用。
解读:
b.intern()
方法最终返回的常量池的地址(对象)
题目二:
注意:
- 编译器在底层会做一个优化,判断创建的常量池对象,是否有引用指向,如果没有,则直接拼接到一起
- String是一个final类,代表不可变的字符序列
- 字符串是不可变的,一个字符串对象一旦被分配,其内容是不可变的
题目三:
public static void main(String[] args) {
String a = "hello";
String b = "abc";
//1.先创建一个 StringBuilder sb = StringBuilder()
//2.执行 sb.append("hello");
//3. sb.append("abc");
//4. Str c =sb.toString();
String c = a + b;
}
String c1 = "ab" + "cd";
常量相加,看的是池String c1 = a + b;
变量相加,是在堆中
小结:
- 底层是
StringBuilder sb = new StringBuilder();
- sb.append(a);
- sb.append(b);
- sb是在堆中,并且append是在原来字符串的基础上追加的
题目四:
三、String常用方法
String类是保存字符串常量的。每次更新都需要重新开辟空间,效率较低,因此java设计者还提供了StringBuilder和 StringBuffer来增强String的功能,并提高效率。[后面我们还会详细介绍StringBuilder 和 StringBuffer]
第一组:
public static void main(String[] args) {
//1.equals比较内容是否相等,区分大小写
String str1 = "hello";
String str2 = "HELLO";
System.out.println(str1.equals(str2));//F
//2.equalsIgnoreCase忽略大小写,比较内容是否相等
System.out.println(str1.equalsIgnoreCase(str2));//T
//3.length获取字符串的长度
System.out.println(str1.length());//5
//4.indexOf 获取字符在字符串对象中第一次出现的索引,索引从 0 开始,如果找不到,返回-1
// 也可以是字符串
System.out.println("l index=" + str1.indexOf('l'));
//5.lastIndexOf 获取字符在字符串中最后一次出现的索引,索引从 0 开始,如果找不到,返回-1,
// 也可以是字符串
System.out.println("l lastIndex=" + str1.lastIndexOf("l"));
//6.substring 截取指定范围的子串
String name = "hello,大龙";
System.out.println(name.substring(6));//大龙,从索引6开始截取,截取所有的内容
System.out.println(name.substring(2,5));//hello,从0开始截取,截取到索引 5-1 的位置
}
第二组:
public static void main(String[] args) {
//1.toUpperCase 转换成大写
String s = "Hello";
System.out.println(s.toUpperCase());//HELLO
//2.toLowerCase
System.out.println(s.toLowerCase());
//3.concat拼接字符串
String s1 = "宝玉";
s1 = s1.concat("林黛玉").concat("together");
System.out.println(s1);
//4.replace 替换字符串中的字符,返回一个替换后的对象,但s1本身并没有被替换
//将s1中所有的宝玉替换成薛宝钗
String s2;
s2 = s1.replace("宝玉","薛宝钗");
System.out.println(s1);//宝玉林黛玉together
System.out.println(s2);//薛宝钗林黛玉together
//5.split 分割字符串, 对于某些分割字符,我们需要 转义比如 | \\
String poem = "锄禾日当午,汗滴禾下土,谁知盘中餐,粒粒皆辛苦";
//老韩解读:
// 1. 以 , 为标准对 poem 进行分割 , 返回一个数组
// 2. 在对字符串进行分割时,如果有特殊字符,需要加入 转义符 \
String[] split = poem.split(",");
poem = "E:\\aaa\\bbb";
split = poem.split("\\\\");
System.out.println("==分割后内容===");
for (int i = 0; i < split.length; i++) {
System.out.println(split[i]);
}
//7.compareTo 比较两个字符串的大小,如果前者大
// 则返回正数,后者大,则返回负数,如果相等,返回 0
// (1)如果长度相同,内容也相同,返回0
// (2)如果长度相同或者不相同,但是在比较适时,可以区分大小
// 返回的是不同值的差值
// (3)如果字符串长度不一样,其他字符都一样,那么就把用前面字符串的长度减去后面字符串的长度
String a = "jack123";// len = 3
String b = "jack";// len = 4
System.out.println(a.compareTo(b));// 返回值是 'c' - 'a' = 2;
String name = "john";
int age = 10 ;
double score = 98.3/3;
char gender = '男';
//将所有的信息都拼接在一个字符串. String info =
String info ="我的姓名是" + name + "年龄是" + age + ",成绩是" + score + "性别是" + gender + "。希望大家喜欢我!";
System.out.println(info);
//注意:
// 保留小数位会进行四舍五入
String formatStr ="我的姓名%5s 年龄%3d ,成绩是%.2f,我的性别%2c.希望大家喜欢我";
String info2 =String.format(formatStr,name,age,score,gender);
System.out.println(info2);
}
四、StringBuffer类
java.lang.StringBuffer
代码可变的字符序列,可以对字符串内容进行增删。
- 很多方法与String相同,但是StringBuffer是可变长度的
- StringBuffer是一个容器
1.String vs StringBuffer
-
String保存的是字符串常量,里面的值不能更改,每次String类的更新实际上是更改地址,效率比较低
private final char value[];
,放在常量池中
-
StringBuffer保存的是字符串变量,里面的值可以更改,每次更新实际上可以更细内容,不用每次更新地址,效率比较高
char[] value;
,放在堆中
String 和StringBuffer相互转换
package com.hspedu.stringbuff_;
/**
* @author DL5O
* @version 1.0
*/
public class StringAndStringBuffer {
public static void main(String[] args) {
//看 String-->StringBuffer
String str = "hello";
//方式1
// 这里只是把str传入到StringBuffer的构造器中,对原来的str没有影响
StringBuffer stringBuffer = new StringBuffer(str);
//方式2
//使用append方法
StringBuffer stringBuffer1 = new StringBuffer();
stringBuffer1 = stringBuffer1.append(str);
//看看 StringBuffer -> String
StringBuffer stringBuffer3 = new StringBuffer("ywl");
//方式1:使用StringBuffer提供的toString方法
String s = stringBuffer3.toString();
//方式2:使用构造器来搞定
String s1 = new String(stringBuffer3);
}
}
2.StringBuffer的常用方法
- 增:
append
- 删:
delete(start,end)
- 改:replace(start,end,string)
- 将start-end间的内容替换掉,不含end
- 查:
indexOf
- 查找子串在字符串第1次出现的索引,如果找不到就返回-1
- 插:
insert
- 获取长度:
length
package com.hspedu.stringbuff_;
/**
* @author DL5O
* @version 1.0
*/
public class StringBufferMethod01 {
public static void main(String[] args) {
StringBuffer stringBuffer = new StringBuffer("hello");
//增
stringBuffer.append(',');//"hello,"
stringBuffer.append("张三丰");//"hello,张三丰"
stringBuffer.append("赵敏").append(100).append(true).append(10.5);
//"hello,张三丰赵敏""
System.out.println(stringBuffer);//hello,张三丰赵敏100true10.5
//删
/*
删除索引>=start && <end 出的字符
解读:删除11~14的字符,但是不包含14
*/
stringBuffer.delete(11,14);//删除100
System.out.println(stringBuffer);//hello,张三丰赵敏true10.5
//改
//使用周芷若替换索引 9~11 的字符,[9,11)
stringBuffer.replace(9,11,"周芷若");
System.out.println(stringBuffer);//hello,张三丰周芷若true10.5
//查找
int indexOf = stringBuffer.indexOf("张三丰");
System.out.println(indexOf);//6
//插入
//老韩解读,在索引为9的位置插入 "赵敏",原来索引为9的内容自动后移
stringBuffer.insert(9,"赵敏");
System.out.println(stringBuffer);//hello,张三丰赵敏周芷若true10.5
System.out.println(stringBuffer.length());
}
}
课堂练习题
package com.hspedu.stringbuff_;
import java.util.Scanner;
/**
* @author DL5O
* @version 1.0
*/
public class StringBufferExercise02 {
public static void main(String[] args) {
/*
输入商品名称和商品价格,要求打印效果示例, 使用前面学习的方法完成:
商品名 商品价
手机 123,564.59 //比如 价格 3,456,789.88
要求:价格的小数点前面每三位用逗号隔开,
思路分析
1.定义一个Scanner类,接受用户的输入
2.使用StringBuffer的insert,需要将String转换成 StringBuffer
3.然后使用相关方法进行字符串的处理
*/
// Scanner sc = new Scanner(System.in);
// System.out.print("请输入商品名:");
// String name = sc.next();
String name = "手机";
// System.out.print("请输入商品价格:");
// String price = sc.next();
String price = "123564123456.69";
StringBuffer sb = new StringBuffer(price);
//先完成最简单的123,564.69
//找到小数点的索引,然后再该位置的前三位插入,号即可
//老师的思路:
for (int index = sb.lastIndexOf(".") -3; index > 0; index -= 3) {
sb = sb.insert(index, ",");
}
System.out.println(sb);
//我的思路:
//int index = sb.lastIndexOf(".");
//index为小数点在字符串中的下标,当下标<=0时,代表到了字符串中的第一个位置,
//就添加字符串完毕,结束循环
/* while (true) {
index = index - 3;
if(index <= 0){
break;
}
sb = sb.insert(index, ",");
}
// sb = sb.insert(index - 3, ",");
System.out.println(sb);
*/
}
}
五、StringBuilder类
- 一个可变的字符序列。此类提供一个与StringBuffer兼容的API,但不保证同步(StringBuilder 不是线程安全)。
- 该类被设计用作 StringBuffer的一个简易替换,用在字符串缓冲区被单个线程使用的时候。如果可能,建议优先采用该类因为在大多数实现中,它比 StringBuffer要快[后面测]。
- 在 StringBuilder上的主要操作是append和 insert方法,可重载这些方法,可以接受任意类型的数据。
注意事项
StingBuilder和StringBuffer均代表可变的字符序列,方法是一样的,所以使用和StringBuffer一样
package com.hspedu.stringbuilder_;
/**
* @author DL5O
* @version 1.0
*/
public class StringBuilder01 {
public static void main(String[] args) {
//1.StringBuilder 继承了AbstractStringBuilder 类
//2.实现了Serializable,说明StringBuilder对象是可以串型化的(该对象可以网络传输)
// 也可以保存到文件
//3.StringBuilder 也是个final类,不能被继承
//4.StringBuilder 对象的字符序列仍然是存放在其父类 AbstractStringBuilder的 byte[] value;
// 因此,字符序列是存放在堆中
//5.StringBuilder 的所有方法没有做互斥的处理,即没有synchronized关键字
// 即推荐在单线程的时候使用
StringBuilder stringBuilder = new StringBuilder();
}
}
六、String、StringBufferStringBuilder
-
StringBuffer和StringBuilder非常相似,他们都代表可变的字符序列,并且方法也一样
-
String:
不可变字符序列,效率低,但是它的复用率高 -
StringBuffer:
可变字符序列,效率高,效率较高(增删改查),线程安全 -
StringBuilder:
可变字符序列,效率最高,线程不安全 -
String使用注意说明:
-
stirng s = "a";
//创建了一个字符串s +="b";
//实际上原来的"a"字符串对象已经丢弃了,现在又产生了一个字符串s+"b"
也就是“ab”
-
如果多次执行,就会有大量的副本留在常量池中,降低效率,影响程序的性能
-
故:如果要对String做大量的修改,就不使用String类
-
总结:
-
如果字符串存在大量的修改操作,一般使用
StringBuffer
或StringBuilder
-
如果字符串存在大量的修改操作,并在单线程的情况,使用
StringBuilder
-
如果字符串存在大量的修改操作,并在多线程的情况,使用
StringBuffer
-
如果我们字符串很少修改,被多个对象引用,使用String,比如配置信息等