视频链接:Java零基础教程
String
字符串是常量,用双引号引起来表示
String实现了Serializable接口:表示字符串是支持序列化的
String实现了Comparable接口:表示String可以比较大小
String 是一个final类,代表不可变的字符序列。String 内部定义了数组 final char[] value 用于存储字符串数据,它们的值在创建之后不能更改
通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值申明在字符串常量池(字符串常量池中是不会存储相同内容的字符串的)中
不可变性:
- 当对字符串重新赋值时,需要重新指定内存区域赋值,不能使用原有的value进行赋值
- 当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值
- 当调用String的replace() 方法修改指定字符或字符串时,也需要重新指定内存区域
String对象的创建
String s1 = "javaEE"; // 方式一:通过字面量的方式。此时的数据声明在方法区中的字符串常量池中
String s2 = new String("javaEE"); // 方式二:通过new + 构造器的方式。此时的数据声明在堆空间中开辟空间以后的地址值
面试题:String s = new String(“abc”); 方式创建对象,在内存中创建了几个对象?
两个:一个是堆空间中new结构,另一个是char[] 对应的常量池中的数据:”abc“
字符串拼接结论:
- 常量与常量的拼接结果在常量池。且常量池中不会存在相同内容的常量
- 只要其中有一个是变量(final声明的为常量,在常量池中),结果就在堆中
- 如果拼接的结果调用intern() 方法,返回值就在常量池中
以此来判断两个字符串拼接后是否 ”==“
面试题:
public class StringTest {
String str = new String("good");
char[] ch = { 't', 'e', 's', 't'};
public void change(String str, char ch[]) {
str = "test ok";
ch[0] = 'b';
}
public static void main(String[] args) {
StringTest ex = new StringTest();
ex.change(ex.str, ex.ch);
System.out.println(ex.str); // good,String是不可变的,新建了一个String地址赋给临时变量
System.out.println(ex.ch); // best
}
}
常量池在不同版本中的位置:
JDK 1.6 | JDK 1.7 | JDK 1.8 |
---|---|---|
方法区(永久代) | 堆 | 方法区(元空间) |
String常用方法一:
方法名 | 说明 |
---|---|
int length() | 返回字符串的长度 return value.length |
char charAt(int index) | 返回某索引处的字符 return value[index] |
boolean isEmpty() | 判断是否是空字符串 return value.length == 0 |
String toLowerCase() | 使用默认语言环境,将String中的所有字符转换为小写 |
String toUpperCase() | 使用默认语言环境,将String中的所有字符转换为大写 |
String trim() | 返回字符串的副本,忽略前导空白和尾部空白 |
boolean equals(Object obj) | 比较字符串的内容是否相同 |
boolean equalsIgnoreCase(String anotherString) | 与equals方法类似,忽略大小写 |
String concat(String str) | 将指定字符串连接到此字符串的结尾。等价于用”+“ |
int compareTo(String anotherString) | 比较两个字符串的大小 |
String substring(int beginIndex) | 返回一个新的字符串,它是此字符串的从beginIndex开始截取到最后的一个子字符串 |
String substring(int beginIndex, int endIndex) | 返回一个新字符串,它是此字符串从beginIndex开始截取到endIndex(不包含)的一个子字符串 |
String常用方法二:
方法名 | 说明 |
---|---|
boolean endsWIth(String suffix) | 测试此字符串是否以指定的后缀结束 |
boolean startsWith(String prefix) | 测试此字符串是否以指定的前缀开始 |
boolean startsWith(String prefix, int toffset) | 测试此字符串从指定索引开始的子字符串是否以指定前缀开始 |
boolean contains(CharSequence s) | 当且仅当此字符串包含指定的char值序列时,返回true |
int indexOf(String str) | 返回指定子字符串在此字符串中第一次出现处的索引 |
int indexOf(String str, int fromIndex) | 返回指定子字符串在此字符串中第一次出现处的索引,从指定的索引开始 |
int lastIndexOf(String str) | 返回指定子字符串在此字符串中最右边出现处的索引 |
int lastIndex(String str, int fromIndex) | 返回指定字符串在此字符串中最后一次出现的索引,从指定的索引开始反向搜索(向左) |
注:indexOf和lastIndexOf方法如果未找到都是返回-1
String常用方法三:
方法名 | 说明 |
---|---|
String replace(char oldChar, char newChar) | 返回一个新的字符串,它是通过用newChar替换此字符串中出现的所有oldChar得到的 |
String replace(CharSequence target, CharSequence replacement) | 使用指定的字面值替换序列替换此字符串所有匹配字面量目标序列的子字符串 |
String replaceAll(String regex, String replacement) | 使用给定的replacement替换此字符串所有匹配给定的正则表达式的子字符串 |
String replaceFirst(String regex, String replacement) | 使用给定的replacement替换此字符串匹配给定的正则表达式的第一个子字符串 |
boolean matches(String regex) | 告知此字符串是否匹配给定的正则表达式 |
String[] split(String regex) | 根据给定正则表达式的匹配拆分此字符串 |
String[] split(String regex, int limit) | 根据匹配给定的正则表达式来拆分此字符串,最多不超过limit个,如果超过了,剩下的全部都放到最后一个元素中 |
String 类型转化
String与基本数据类型包装类的转化 | - |
---|---|
toString(); | Integers.parseInt(); |
String与字符数组的转化 | - |
---|---|
str.toCharArray(); | new String(charArray); |
String与字节数组的转化 | - |
---|---|
str.getBytes(); // 使用默认的字符集进行转换 | new Sring(byteArray); |
str.getBytes(”utf-8“); // 指定字符集进行转换 | new Sring(byteArray, “utf-8”); |
字节(数值)编码 <–> 解码 字符(符号)
utf-8 编码一个汉字占三个字节
gbk 编码一个汉字占两个字节
解码时,要求解码使用的字符集必须与编码时使用的字符集一致,否则会出现乱码
StringBuffer
java.lang.StringBuffer代表可变的字符序列,JDK1.0中声明,可以对字符串内容进行增删,此时不会产生新的对象
很多方法和String相同
作为参数传递时,方法内部可以改变值
String、StringBuffer、StringBuilder三者的异同?
- | String | StringBuffer | StringBuilder |
---|---|---|---|
版本 | JDK 1.0 | JDK 1.0 | JDK 5.0 |
是否可变 | 不可变的字符序列 | 可变的字符序列 | 可变的字符序列 |
线程安全/拼接效率 | 线程不安全,效率最低 | 线程安全,效率低 | 线程不安全,效率高 |
存储结构 | final char[] | char[] | char[] |
源码分析:
String str = new String(); // new char[0];
String str1 = new String("abc"); // new char[] {'a', 'b', 'c'};
StringBuffer sb1 = new StringBuffer(); // new char[16]; 底层创建了一个长度为16的数组
sb1.append('a'); // value[0] = 'a';
sb1.append('b'); // value[1] = 'b';
StringBuffer sb2 = new StringBuffer("abc"); // char[] value = new char["abc".length() + 16];
// 问题 1. sb2.length() ? 3 返回的是count值
// 问题 2. 扩容问题,如果要添加的数据底层数组盛不下了,那就需要扩容底层的数组
// 默认情况下扩容为 value.length << 1 + 2,同时将原来数组中的元素复制到新的数组中
// 指导意义:开发中建议大家使用:StringBuffer(int capacity)
StringBuffer和StringBuilder的主要区别还在于StringBuffer里的方法都加了synchronized关键字
StringBuffer类的常用方法
方法名 | 说明 |
---|---|
StringBuffer append(xxx) | 提供了很多的append() 方法,用于进行字符串拼接 |
StringBuffer delete(int start, int end) | 删除指定位置的内容 |
StringBuffer replace(int start, int end, String str) | 把[start, end) 位置替换为str |
StringBuffer insert(int offset, xxx) | 在指定位置插入xxx |
StringBuffer reverse() | 把当前字符序列逆转(String没有此方法) |
当append和insert时,如果原来value数组长度不够,可扩容
如上这些方法支持方法链操作
方法链原理:return this;
@Override
public StringBuilder append(String str) {
super.append(str);
return this;
}
此外还定义了如下方法:
方法 |
---|
public int indexOf(String str) |
public String substring(int start, int end) |
public int length() |
public char charAt(int n) |
public void setCharAt(int n, char ch) |
面试题:获取两个字符串最大相同子串(首个)
/**
* 获取两个字符串中最大相同子串(首个)
*/
public static String getMaxSameString(String str1, String str2) {
if (str1 != null && str2 !=null) {
String maxStr = (str1.length() >= str2.length()) ? str1 : str2;
String minStr = (str1.length() < str2.length()) ? str1 : str2;
int length = minStr.length();
for (int i = 0; i < length; i++) {
for (int x=0, y = length - i;y <= length;x++, y++) {
String subStr = minStr.substring(x, y);
if (maxStr.contains(subStr)) {
return subStr;
}
}
}
}
return null;
}
面试题:StringBuffer拼接null
String str = null;
StringBuffer sb = new StringBuffer();
sb.append(str);
System.out.println(sb.length()); // 4
System.out.println(sb); // "null"
StringBuffer sb1 = new StringBuffer(str); // 空指针异常
System.out.println(sb1);