String类

1.创建字符串

//方式一
String str = "hello java";
//方式二
String str2 = new String("hello java");
//方式三
char[] arr = {'a','b','c'}; 
String str3 = new String(arr);

“hello” 这样的字符串字面值常量,类型也是String。
String也是引用类型,String str = “Hello”;这种代码内存布局如下:
String类_第1张图片
由于String 是引用类型,对于下列代码:

String str1 = "Hello";
String str2 = str1;

内存布局如图
String类_第2张图片

2.字符串比较相等

int x = 10 ;
int y = 10 ;
System.out.println(x == y); 
// 执行结果
true

如果说现在在String类对象上使用 == ?

String str1 = "Hello";
String str2 = "Hello"; 
System.out.println(str1 == str2); 
// 执行结果
true
String str1 = new String("Hello");
String str2 = new String("Hello");
System.out.println(str1 == str2);
// 执行结果
false

String类_第3张图片
通过 String str1 = new String(“Hello”); 这样的方式创建的 String 对象相当于再堆上另外开辟了空间来存储
“Hello” 的内容, 也就是内存中存在两份 “Hello”.
String 使用 == 比较并不是在比较字符串内容, 而是比较两个引用是否是指向同一个对象.
Java 中要想比较字符串的内容,必须采用String类提供的equals方法

String str1 = new String("Hello");
String str2 = new String("Hello");
System.out.println(str1.equals(str2));
// System.out.println(str2.equals(str1)); // 或者这样写也行
// 执行结果
true

3.字符串常量池

a) 直接赋值

String str1 = "hello" ; 
String str2 = "hello" ; 
String str3 = "hello" ; 
System.out.println(str1 == str2); // true 
System.out.println(str1 == str3); // true 
System.out.println(str2 == str3); // true

String类的设计使用了共享设计模式
在JVM底层实际上会自动维护一个对象池(字符串常量池)

如果现在采用了直接赋值的模式进行String类的对象实例化操作,那么该实例化对象(字符串内容)将自动保存
到这个对象池之中.
如果下次继续使用直接赋值的模式声明String类对象,此时对象池之中如若有指定内容,将直接进行引用
如若没有,则开辟新的字符串对象而后将其保存在对象池之中以供下次使用
(b)构造方法

String str = new String("hello");

这样的做法有两个缺点:
1.如果使用String构造方法就会开辟两块堆内存空间,并且其中一块堆内存将成为垃圾空间(字符串常量 “hello” 也
是一个匿名对象, 用了一次之后就不再使用了, 就成为垃圾空间, 会被 JVM 自动回收掉).
2.字符串共享问题. 同一个字符串可能会被存储多次,比较浪费空间。

我们可以使用 String 的 intern 方法来手动把 String 对象加入到字符串常量池中

// 该字符串常量并没有保存在对象池之中
String str1 = new String("hello") ; 
String str2 = "hello" ; 
System.out.println(str1 == str2); 
// 执行结果
false 
String str1 = new String("hello").intern() ; 
String str2 = "hello" ; 
System.out.println(str1 == str2); 
// 执行结果
true

4.理解字符串不可变

字符串是一种不可变对象. 它的内容不可改变.
String 类的内部实现也是基于 char[] 来实现的, 但是 String 类并没有提供 set 方法之类的来修改内部的字符数组.

String str = "hello" ; 
str = str + " world" ; 
str += "!!!" ; 
System.out.println(str); 
// 执行结果
hello world!!!

形如 += 这样的操作, 表面上好像是修改了字符串, 其实不是.
为什么 String 要不可变?(不可变对象的好处是什么?) (选学)

  1. 方便实现字符串对象池. 如果 String 可变, 那么对象池就需要考虑何时深拷贝字符串的问题了.
  2. 不可变对象是线程安全的.
  3. 不可变对象更方便缓存 hash code, 作为 key 时可以更高效的保存到 HashMap 中.

5.字符、字节与字符串

5.1字符与字符串

字符串内部包含一个字符数组,String 可以和 char[] 相互转换.

No 方法名称 类型 描述
1 public String(char value[]) 构造 将字符数组中的内容变为字符串
2 public String(char value[],int offset,int count) 构造 将部分字符数组中的内容变为字符串
3 public char charAt(int index) 普通 取得索引位置的字符,索引从0开始
4 publc char[] toCharArry() 普通 将字符变为字符数组返回

5.2字节与字符串

字节常用于数据传输以及编码转换的处理之中,String 也能方便的和 byte[] 相互转换.

No 方法名称 类型 描述
1 public String(byte bytes[]) 构造 将字节数组中的内容变为字符串
2 public String(byte bytes[],int offset,int length) 构造 将部分字节数组中的内容变为字符串
3 public byte[] getBytes() 普通 将字符串以字节数组的形式返回

6.字符串常见操作

6.1字符串比较

No 方法名称 类型 描述
1 public boolean equals(Object anObject) 普通 区分大小写的比较
2 public boolean equalsIgnoreCase(String anotherString) 普通 不区分大小写的比较
3 public int compareTo(String anotherString) 普通 比较两个字符串大小关系
System.out.println("A".compareTo("a")); // -32 
System.out.println("a".compareTo("A")); // 32 
System.out.println("A".compareTo("A")); // 0 
System.out.println("AB".compareTo("AC")); // -1 
System.out.println("刘".compareTo("杨"));

compareTo()是一个可以区分大小关系的方法,是String方法里是一个非常重要的方法。

6.2字符串查找

No 方法名称 类型 描述
1 public boolean contains(charSequence s) 普通 判断一个字符串是否存在
2 public int indexOf(String str) 普通 从头开始查找指定字符串的位置,查到了返回位置的开始索引,如果查不到返回 -1
3 public int indexOf(String str,int fromIndex) 普通 从指定位置开始查找子字符串位置
4 public int lastIndexOf(String str) 普通 由后向前查找子字符串位置
5 public int lastIndexOf(String str,int fromIndex) 普通 从指定位置由后向前查找
6 public boolean startsWith(String prefix) 普通 判断是否以指定字符串开头
7 public boolean startsWith(String prefix,int toffest) 普通 从指定位置开始判断是否以指定字符串开头
8 public boolean endsWith(String suffix) 普通 判断是否以指定字符串结尾

6.3字符串替换

No 方法名称 类型 描述
1 public String replaceAll(String regex,String replacement) 普通 替换所有的指定内容
2 public String replaceFirst(String regex,String replacement) 普通 替换首个内容

6.4字符串拆分

No 方法名称 类型 描述
1 public String[] split(String regex) 普通 将字符串全部拆分
2 public String[] split(String regex,int limit) 普通 将字符串部分拆分,该数组长度就是limit极限

6.5字符串截取

No 方法名称 类型 描述
1 public String substring(int beginIndex) 普通 从指定索引截取到结尾
2 public String substring(int beginIndex,int endIndex) 普通 截取部分内容

7.StringBuffer and StringBuilder

通常来讲String的操作比较简单,但是由于String的不可更改特性,为了方便字符串的修改,提供StringBuffer和StringBuilder类。
StringBuffer和StringBuilder 大部分功能是相同的。
在String中使用"+"来进行字符串连接,但是这个操作在StringBuffer类中需要更改为append()方法:

public synchronized stringBuffer append(各种类型数据 b)

StringBuffer采用同步处理,属于线程安全操作;而StringBuilder采用异步处理,属于线程不安全操作

public class Test{ 
 public static void main(String[] args) { 
 StringBuffer sb = new StringBuffer(); 
 sb.append("Hello").append("World"); 
 fun(sb); 
 System.out.println(sb); 
 } 
 public static void fun(StringBuffer temp) { 
 temp.append("\n"); 
 } 
}

String和StringBuffer最大的区别在于:String的内容无法修改,而StringBuffer的内容可以修改。频繁修改字符串的情况考虑使用StingBuffer。
String和StringBuffer类不能直接转换。如果要想互相转换,可以采用如下原则:
String变为StringBuffer:利用StringBuffer的构造方法或append()方法。
StringBuffer变为String:调用toString()方法。

你可能感兴趣的:(String类)