String 类
实例化
String 本身包装的是一个数组,并且其有两种对象的实例化形式:直接赋值、构造方法实例化。
public class StringDemo {
public static void main(String args[]){
// 直接赋值
String str1 = "Hello,World";
// 构造方法实例化
String str2 = new String("Hello,World");
String str3 = "Hello,World";
}
}
直接赋值实例化
直接赋值实例化 String 的形式还可以实现同一个字符串对象数据的共享,即 str1 和 str3 指向同一块堆内存空间。
之所以有这样的特点,是因为JVM中提供了一个专门的“字符串池”,本质上就是“字符串数组”。
在采用直接赋值的处理过程中,对于字符串而言可以实现池数据的自动保存,如果再有相同数据定义时,可以减少对象的产生,提升操作性能。
// TODO 那当某个引用对字符串对象进行更改操作后,另一个引用显示会不同吗?
答:由于 String 的不可修改性,某个引用指向的字符串一旦发生修改,就会为该引用重新开辟一个堆内存,用来存储修改后的字符串。因此,另一个引用依然指向原字符串,显示不变。
构造方法实例化
使用构造方法实例化 String 对象时,不会自动保存到字符串池的处理。
面试题:解释 String 类两种对象实例化方式的区别?
- 直接赋值:只会产生一个实例化对象,并且自动保存到字符串池中,以实现该字符串实例的重用。
- 构造方法:会产生两个实例化对象,并且不会自动入池,无法实现对象重用,但是可以利用 intern() 方法手工入池处理。
字符串比较
String 比较中 “==” 与 equals() 的区别:
- “==”:进行的是数值比较,如果用于对象比较上,比较的是两个内存的地址数值,即比较两个栈内存的地址值是否指向同一块堆内存实例。
- equals():是类所提供的一个比较方法,可以直接进行字符串“内容”的比较判断。
public class StringDemo {
public static void main(String args[]){
// 直接赋值
String str1 = "Hello,World";
// 构造方法实例化,新的对象
String str2 = new String("Hello,World");
// 直接将 str3 指向堆内存中已经存在的 str1 指向的实例
String str3 = "Hello,World";
System.out.println(str1 == str2); // 输出 false
System.out.println(str1 == str3); // 输出 true
}
}
字符串常量
- 一个字符串常量
"Hello,World!"
,本质上就是一个 String 类的匿名对象。 - 所谓的字符串的直接赋值
String str1 = "Hello,World!";
,描述的就是将一个匿名对象设置一个具体的引用,即将堆中内存首地址放入栈内存。
关于字符串对象相等判断的技巧:
之后进行项目开发的时候,如果现在某些数据是有用户输入,并且要求这些数据为一个指定内容的情况下,建议将字符串常量写在前面。
方法一: 将用户输入的字符串写在前面
此时,若当用户无输入时,会出现异常。
public class test{
public static void main(String args[]){
String input = null; // 用户输入的内容
System.out.println(input.equals("yes"));
}
}
// Exception in thread "main" java.lang.NullPointerException
方法二:将字符串常量写在前面
public class test{
public static void main(String args[]){
String input = null; // 用户输入的内容
System.out.println("yes".equals(input));
}
}
// true
因为,equals() 方法中有可以回避 null 的判断,所以如果将字符串常量写在前面,那么调用 equals() 方法时,永远不会出现 “NullPointerException”。字符串是一个匿名对象,匿名对象一定是开辟好堆内存空间的。
String 对象常量池
静态常量池
当给出的内容全部都是常量数据(字符串的常量是匿名对象),所以最终在程序加载的时候会自动帮助开发者处理好相应的连接,再和对象池中进行比较,发现相同,故指向同一个堆内存。
String str1 = "yes";
String str2 = "ye" + "s";
System.out.println(str1 == str2); // true
运行时常量池
程序再加载的时候,遇到变量 strs
并不确定其中是什么内容,变量的内容是可以修改的,所以它不认为是与 str2
就是最终的结果。
此种情况,会出现再当字符串拼接中,有用户输入的变量时,应该使用 equals() 方法区比较。
String str1 = "yes";
String strs = "s";
String str2 = "ye" + strs;
System.out.println(str1 == str2); // false
字符串常用方法
字符串与字符数组
字符串与字节数组
字符串和字节数组进行转化的目的是为了进行二进制的数据传输,或者是进行编码转换。
字符串比较
equals() 方法在进行比较时,是对大小写进行区分的。
equalsIgnoreCase() 忽略大小写
compareTo() 进行字符串比较,并且计算出两者对应于ASCLL码的差值。
compareToIgnoreCase() 忽略大小写
字符串查找
从完整的字符串之中查找子字符串的存在
字符串替换
在开发、设计中,替换很重要。可使用正则表达 regex
String str = "HelloWorld";
System.out.println(str.replaceAll("l","x")); // "HexxoWorxd"
System.out.println(str.replaceFirst("l","x")); // "HexloWorld"
字符串拆分
字符串截取
在实际的开发中,有些截取所需的下标索引,是通过字符串查找函数 indexof() 来获得的。
字符串格式化
其他操作方法
字符串定义的时候,""
和 null
不是一个概念,一个表示有实例化对象,一个表示没有进行实例化操作,isempty()
主要判断的是空内容,而不是null。