String 类的理解和创建对象
- 1,String 对象用于保存字符串,也就是一组字符数列
- 2,字符串常量对象是用双引号括起的字符序列。例如:“你好”、“12.97”、“boy”等
- 3,字符串的字符使用Unicode字符编码,一个字符(不区分字母还是汉字)占两个字节
- 4,String类较常用的构造器
String s1 = new String();
String s2 = new String(String original);
String s3 = new String(char[] a);
String s4 = new String(char[] a,int startIndex,int count);
- 5,String 类实现了接口 Serializable【String 可以串行化:可以在网络传输】
接口 Comparable [String 对象可以比较大小] - 6,String 是 final 类,不能被其他的类继承
- 7,String 有属性 private final char value[]; 用于存放字符串内容
- 8,一定要注意:value 是一个 final 类型, 不可以修改(需要功力):即 value 不能指向新的地址,但是单个字符内容是可以变化
public class String01 {
public static void main(String[] args) {
String name = "jack";
name = "tom";
final char[] value = {'a','b','c'};
char[] v2 = {'t','o','m'};
value[0]='H';//单个字符内容是可以变化
//value = v2; 不可以修改value地址
}
}
创建String对象的两种方式
- 方式一:直接赋值 String s = “zl”;
- 方式二:调用构造器 String s = new String(“zl”);
- 两种创建String对象的区别
1,方式一:先从常量池中查看是否有“zl”数据空间,如果有,直接访问;如果没有,则重新创建,然后指向。s最终指向的是常量池的空间地址
2,方式二:现在堆中创建空间,里面放置了value属性,指向常量池的zl空间。如果常量池中没有“zl”,重新创建,如果有,直接通过value指向。s最终指向的是堆中的空间地址
字符串的特性
- String是一个final类,代表不可变的字符序列
- 字符串是不可变的。一个字符串对象一旦被分配,其内容是不可变的
String类的常见方法
说明
String类是保存字符串常量的,每次更新都需要重新开辟空间,效率极低。
String类的常见方法一览
- equals//区分大小写,判断内容是否相等
- equalsIgnoreCase//忽略大小写,判断内容是否相等
- length//获取字符的个数,字符串的长度
- indexOf//获取字符在字符串中第1次出现的索引,索引从0开始,如果找不到,返回-1
- lastIndexOf//获取字符在字符串中最后1次出现的索引,索引从0开始,如果找不到,返回-1
- substring//截取指定范围的子串
- trim//去前后空格
- charAt//获取某索引处的字符,注意不能使用Str[index]这种方式
- toUpperCase
- toLowerCase
- concat
- replace 替换字符串中的字符
- split 分隔字符串,对于某些分隔字符,我们需要转义 比如|\等
- compareTo //比较两个字符串的大小
- toCharArray//转换成字符数组
- format//格式字符串,%s字符串 %c字符 %d整形 %.2f浮点型
public class String01 {
public static void main(String[] args) {
//1. equals 比较内容是否相同,区分大小写
String str1 = "hello";
String str2 = "Hello";
System.out.println(str1.equals(str2));//T
// 2.equalsIgnoreCase 忽略大小写的判断内容是否相等
String username = "johN";
if ("john".equalsIgnoreCase(username)) {
System.out.println("Success!");//输出
} else {
System.out.println("Failure!");//不输出
}
// 3.length 获取字符的个数,字符串的长度
System.out.println("zl".length());
// 4.indexOf 获取字符在字符串对象中第一次出现的索引,索引从 0 开始,如果找不到,返回-1
String s1 = "wer@terwe@g";
int index = s1.indexOf('@');
System.out.println(index);// 3
System.out.println("weIndex=" + s1.indexOf("we"));//0
// 5.lastIndexOf 获取字符在字符串中最后一次出现的索引,索引从 0 开始,如果找不到,返回-1
s1 = "wer@terwe@g@";
index = s1.lastIndexOf('@');
System.out.println(index);//11
System.out.println("ter 的位置=" + s1.lastIndexOf("ter"));//4
// 6.substring 截取指定范围的子串
String name = "hello,张三";
//下面 name.substring(6) 从索引 6 开始截取后面所有的内容
System.out.println(name.substring(6));//截取后面的字符
//name.substring(0,5)表示从索引 0 开始截取,截取到索引 5-1=4 位置
System.out.println(name.substring(2,5));//llo
// 1.toUpperCase 转换成大写
String s = "heLLo";
System.out.println(s.toUpperCase());//HELLO
// 2.toLowerCase 转换成小写
System.out.println(s.toLowerCase());//hello
// 3.concat 拼接字符串
String s2 = "宝玉";
s2 = s2.concat("林黛玉").concat("薛宝钗").concat("together");
System.out.println(s2);//宝玉林黛玉薛宝钗 together
// 4.replace 替换字符串中的字符
s2 = "宝玉 and 林黛玉 林黛玉 林黛玉";
//在 s2中,将 所有的 林黛玉 替换成薛宝钗
// s2.replace() 方法执行后,返回的结果才是替换过的. // 注意对 s2 没有任何影响
String s11 = s2.replace("宝玉", "jack");
System.out.println(s2);//宝玉 and 林黛玉 林黛玉 林黛玉
System.out.println(s11);//jack and 林黛玉 林黛玉 林黛玉
// 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]);
}
// 6.toCharArray 转换成字符数组
s = "happy";
char[] chs = s.toCharArray();
for (int i = 0; i < chs.length; i++) {
System.out.println(chs[i]);
}
// 7.compareTo 比较两个字符串的大小,如果前者大,
// 则返回正数,后者大,则返回负数,如果相等,返回 0
// (1) 如果长度相同,并且每个字符也相同,就返回 0
// (2) 如果长度相同或者不相同,但是在进行比较时,可以区分大小
// 就返回 if (c1 != c2) {
// return c1 - c2;
// }
// (3) 如果前面的部分都相同,就返回 str1.len - str2.len
String a = "jcck";// len = 3
String b = "jack";// len = 4
System.out.println(a.compareTo(b)); // 返回值是 'c' - 'a' = 2 的值
// 8.format 格式字符串
/* 占位符有:
* %s 字符串 %c 字符 %d 整型 %.2f 浮点型
*
*/
String name1 = "john";
int age = 10;
double score = 56.857;
char gender = '男';
//将所有的信息都拼接在一个字符串.
String info =
"我的姓名是" + name + "年龄是" + age + ",成绩是" + score + "性别是" + gender +"希望大家喜欢我";
System.out.println(info);
//1. %s , %d , %.2f %c 称为占位符
//2. 这些占位符由后面变量来替换
//3. %s 表示后面由 字符串来替换
//4. %d 是整数来替换
//5. %.2f 表示使用小数来替换,替换后,只会保留小数点两位, 并且进行四舍五入的处理
//6. %c 使用 char 类型来替换
String formatStr = "我的姓名是%s 年龄是%d,成绩是%.2f 性别是%c.希望大家喜欢我!";
String info2 = String.format(formatStr, name, age, score, gender);
System.out.println("info2=" + info2);
}
}
StringBuffer 类
基本介绍
- java.lang.StringBuffer代表可变的字符序列,可以对字符串内容进行增删
- 很多方法与String相同,但StringBuffer是可变长度的
- StringBuffer是一个容器
String VS StringBuffer
- String保存的是字符串常量,里面的值不能更改,每次String类的更新实际上就是更改地址,效率较低
- StringBuffer保存的是字符串变量,里面的值可以更改;StringBuffer的更新实际上可以更新内容,不用每次更新地址,效率较高 //char[] value 放在堆中
String 和 StringBuffer 相互转换
public class StringAndStringBuffer {
public static void main(String[] args) {
//String->StringBuffer
String str = "hello";
//方式 1 使用构造器
//注意: 返回的才是 StringBuffer 对象,对 str 本身没有影响
StringBuffer stringBuffer = new StringBuffer(str);
//方式 2 使用的是 append 方法
StringBuffer stringBuffer1 = new StringBuffer();
stringBuffer1 = stringBuffer1.append(str);
//StringBuffer->String
StringBuffer stringBuffer2 = new StringBuffer("zl");
//方式 1 使用 StringBuffer 提供的 toString 方法
String string = stringBuffer2.toString();
//方式 2: 使用构造器来搞定
String s = new String(stringBuffer2);
}
}
StringBuffer 类常见方法
- 增append
- 删delete
- 改replace
- 插insert
public class StringAndStringBuffer {
public static void main(String[] args) {
StringBuffer s = new StringBuffer("hello");
//增
s.append(',');// "hello,"
s.append("张三丰");//"hello,张三丰"
s.append("赵敏").append(100).append(true).append(10.5);//"hello,张三丰赵敏 100true10.5" System.out.println(s);//"hello,张三丰赵敏 100true10.5"
//删
/*
* 删除索引为>=start && <end 处的字符
* 解读: 删除 11~14 的字符 [11, 14)
*/
s.delete(11, 14);
System.out.println(s);//"hello,张三丰赵敏 true10.5"
//改
//使用 周芷若 替换 索引 9-11 的字符 [9,11)
s.replace(9, 11, "周芷若");
System.out.println(s);//"hello,张三丰周芷若 true10.5"
//查找指定的子串在字符串第一次出现的索引,如果找不到返回-1
int indexOf = s.indexOf("张三丰");
System.out.println(indexOf);//6
//插
//在索引为 9 的位置插入 "赵敏",原来索引为 9 的内容自动后移
s.insert(9, "赵敏");
System.out.println(s);//"hello,张三丰赵敏周芷若 true10.5"
//长度
System.out.println(s.length());//22
System.out.println(s);
}
}
StringBuilder 类
基本介绍
- 一个可变的字符序列。此类提供一个与StringBuffer先容的API,但不保证同步(StringBuilder 不是线程安全)。该类被设计用作StringBuffer的一个简易替换,用在字符串缓冲区被单个线程使用时,如果可能,建议优先采用该类,因为在大多数实现中,它比StringBuffer要快
- 在StringBuilder上的主要操作是append和insert方法,可重载这些方法,以接收任意类型的数据
- StringBuffer 的直接父类 是 AbstractStringBuilder
- StringBuffer 实现了 Serializable, 即 StringBuffer 的对象可以串行化
- 在父类中 AbstractStringBuilder 有属性 char[] value,不是 final,该 value 数组存放 字符串内容,引出存放在堆中的
- StringBuffer 是一个 final 类,不能被继承
- 因为 StringBuffer 字符内容是存在 char[] value, 所有在变化(增加/删除),不用每次都更换地址(即不是每次创建新对象), 所以效率高于 String
StringBuilder 常用方法
- StringBuilder和StringBuilder 均代表可变的字符序列,方法是一样的
- StringBuilder 的方法,没有做互斥的处理,即没有 synchronized 关键字,因此在单线程的情况下使用
String、StringBuffer 和 StringBuilder 的比较
- StringBuffer和StringBuilder 非常相似,均代表可变的字符序列,而且方法也一样
- String:不可变字符序列,效率低,但是复用率高
- StringBuffer:可变字符序列,效率较高(增删)、线程安全
- StringBuilder:可变字符序列,效率最高,线程不安全
- String使用注意说明
string s = “a”;//创建了一个字符串
s+=“b”;//实际上原来的“s”字符串对象已经丢弃了,现在又产生一个字符串s+“b”(也就是“ab”)。
如果多次执行这些改变串内容的操作,会导致大量副本字符串对象存留在内存中,降低效率。如果这样的操作放到循环中,会极大影响程序得性能
**结论:**如果我们对String做大量修改,不要用String
String、StringBuffer 和 StringBuilder 的效率测试
StringBuilder > StringBuffer > String
public class StringAndStringBuffer {
public static void main(String[] args) {
long startTime = 0L;
long endTime = 0L;
StringBuffer buffer = new StringBuffer("");
startTime = System.currentTimeMillis();
for (int i = 0; i < 80000; i++) {//StringBuffer 拼接 20000 次
buffer.append(String.valueOf(i));
}
endTime = System.currentTimeMillis();
System.out.println("StringBuffer 的执行时间:" + (endTime - startTime));
StringBuilder builder = new StringBuilder("");
startTime = System.currentTimeMillis();
for (int i = 0; i < 80000; i++) {//StringBuilder 拼接 20000 次
builder.append(String.valueOf(i));
}
endTime = System.currentTimeMillis();
System.out.println("StringBuilder 的执行时间:" + (endTime - startTime));
String text = "";
startTime = System.currentTimeMillis();
for (int i = 0; i < 80000; i++) {//String 拼接 20000
text = text + i;
}
endTime = System.currentTimeMillis();
System.out.println("String 的执行时间:" + (endTime - startTime));
}
}
结果