String类

创建字符串

常见的构造 String 的方式:

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

String 也是引用类型. String str = “Hello”; 这样的代码内存布局如下:
引用只是在栈上开辟了一小块内存空间保存一个地址,指向堆里的对象.
在这里插入图片描述

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

内存布局:
在这里插入图片描述

字符串比较相等

现在有两个int型变量,判断其相等可以使用 == 完成

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

看看两个代码的内存布局:
代码一:
在这里插入图片描述
此时如 “Hello” 这样的字符串常量是在字符串常量池中
代码二:
在这里插入图片描述
通过 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

equals 使用注意事项
现在需要比较 str 和 “Hello” 两个字符串是否相等,有两种方法:

String str = new String("Hello");
// 方式一
System.out.println(str.equals("Hello"));
// 方式二
System.out.println("Hello".equals(str));

更推荐第二种写法,一旦 str 是 null, 方式一的代码会抛出异常, 而方式二不会。

字符串常量池

直接赋值

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类对象,此时对象池之中如若有指定内容,将直接进行引用如若没有,则开辟新的字符串对象而后将其保存在对象池之中以供下次使用;

采用构造方法

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

在这里插入图片描述
请解释String类中两种对象实例化的区别
1.直接赋值:只会开辟一块堆内存空间,并且该字符串对象可以自动保存在对象池中以供下次使用。
2. 构造方法:会开辟两块堆内存空间,不会自动保存在对象池中,可以使用intern()方法手工入池。

理解字符串不可变

String 类的内部实现也是基于 char[] 来实现的, 但是 String 类并没有提供 set 方法之类的来修改内部的
字符数组。

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

在这里插入图片描述

形如 += 这样的操作, 表面上好像是修改了字符串, 其实不是,+= 之后 str 打印的结果却是变了, 但是不是 String 对象本身发生改变, 而是 str 引用到了其他的对象。是引用中存的地址改变了
如果实在需要修改字符串
常见办法: 借助原字符串, 创建新的字符串

String str = "Hello";
str = "h" + str.substring(1);
System.out.println(str);
// 执行结果
hello

str.substring() 的用法:截取掉str从首字母起长度为startIndex的字符串,将剩余字符串赋值给str;

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

在这里插入图片描述

在这里插入图片描述

字符, 字节与字符串

字符与字符串

字符串内部包含一个字符数组,String 可以和 char[] 相互转换
在这里插入图片描述
代码示例: 获取指定位置的字符

String str = "hello" ; 
System.out.println(str.charAt(0));  // 下标从 0 开始
// 执行结果
h
System.out.println(str.charAt(10));
// 执行结果
//产生 StringIndexOutOfBoundsException 异常
//字符串截取下标越界

代码示例: 字符串与字符数组的转换

String str = "helloworld" ; 
// 将字符串变为字符数组
char[] data = str.toCharArray() ; 
for (int i = 0; i < data.length; i++) {
    System.out.print(data[i]+" ");
}
// 字符数组转为字符串
System.out.println(new String(data)); // 全部转换
System.out.println(new String(data,5,5)); // 部分转换

代码示例: 给定字符串一个字符串, 判断其是否全部由数字所组成
思路: 将字符串变为字符数组而后判断每一位字符是否是" 0 “~”‘9’"之间的内容,如果是则为数字.

public static void main(String[] args) {
 String str = "1a23456" ; 
 System.out.println(isNumber(str)? "字符串由数字所组成!" : "字符串中有非数字成
员!");
 } 
 public static boolean isNumber(String str) {
 char[] data = str.toCharArray() ; 
 for (int i = 0; i < data.length; i++) {
 if (data[i]<'0' || data[i]>'9') {
 return false ; 
 }
 }
 return true ; 
 }

字节与字符串

字节常用于数据传输以及编码转换的处理之中,String 也能方便的和 byte[] 相互转换
在这里插入图片描述
代码示例: 实现字符串与字节数组的转换处理

String str = "helloworld" ; 
// String 转 byte[]
byte[] data = str.getBytes() ; 
for (int i = 0; i < data.length; i++) { 
    System.out.print(data[i]+" ");
}
// byte[] 转 String
System.out.println(new String(data));

小结

那么何时使用 byte[], 何时使用 char[] 呢?
byte[] 是把 String 按照一个字节一个字节的方式处理, 这种适合在网络传输, 数据存储这样的场景下使用. 更适合针对二进制数据来操作.
char[] 是吧 String 按照一个字符一个字符的方式处理, 更适合针对文本数据来操作, 尤其是包含中文的时候。

字符串常见操作

在这里插入图片描述
在这里插入图片描述

字符串比较

代码示例: 不区分大小写比较

String str1 = "hello" ; 
String str2 = "Hello" ; 
System.out.println(str1.equals(str2)); // false  区分
System.out.println(str1.equalsIgnoreCase(str2)); // true  不区分

在String类中compareTo()方法是一个非常重要的方法,该方法返回一个整型,该数据会根据大小关系
返回三类内容:
1.相等:返回0.
2. 小于:返回内容小于0.
3. 大于:返回内容大于0
范例:观察compareTo()比较

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方法里是一个非常重要的方法。

字符串查找

在这里插入图片描述
代码示例: 字符串查找,最好用最方便的就是contains()

String str = "helloworld" ; 
System.out.println(str.contains("world")); // true

代码示例: 使用indexOf()方法进行位置查找

String str = "helloworld" ; 
System.out.println(str.indexOf("world")); // 5,w开始的索引
System.out.println(str.indexOf("bit")); // -1,没有查到
if (str.indexOf("hello") != -1) {
    System.out.println("可以查到指定字符串!");
}

使用indexOf()需要注意的是,如果内容重复,它只能返回查找的第一个位置
代码示例: 使用indexOf()的注意点

String str = "helloworld" ; 
System.out.println(str.indexOf("l")); // 2
System.out.println(str.indexOf("l",5)); // 8
System.out.println(str.lastIndexOf("l")); // 8

在进行查找的时候往往会判断开头或结尾。
代码示例: 判断开头或结尾

String str = "**@@helloworld!!" ; 
System.out.println(str.startsWith("**")); // true
System.out.println(str.startsWith("@@",2)); // ture
System.out.println(str.endsWith("!!")); // true

字符串替换

使用一个指定的新的字符串替换掉已有的字符串数据,可用的方法如下:
在这里插入图片描述
代码示例: 字符串的替换处理

String str = "helloworld" ; 
System.out.println(str.replaceAll("l", "_"));
System.out.println(str.replaceFirst("l", "_"));

注意事项: 由于字符串是不可变对象, 替换不修改当前字符串, 而是产生一个新的字符串.

字符串拆分

可以将一个完整的字符串按照指定的分隔符划分为若干个子字符串
在这里插入图片描述
代码示例: 实现字符串的拆分处理

String str = "hello world hello bit" ; 
String[] result = str.split(" ") ; // 按照空格拆分
for(String s: result) {
    System.out.println(s);
}

代码示例: 字符串的部分拆分

String str = "hello world hello bit" ; 
String[] result = str.split(" ",2) ; 
for(String s: result) {
    System.out.println(s);
}

代码示例: 拆分IP地址

String str = "192.168.1.1" ; 
String[] result = str.split("\\.") ; 
for(String s: result) {
    System.out.println(s);
}

注意事项:
1.字符"|","*","+“都得加上转义字符,前面加上”".
2. 而如果是"",那么就得写成"\".
3. 如果一个字符串中有多个分隔符,可以用"|"作为连字符

代码示例: 多次拆分

String str = "name=zhangsan&age=18" ; 
String[] result = str.split("&") ;
for (int i = 0; i < result.length; i++) {
    String[] temp = result[i].split("=") ; 
    System.out.println(temp[0]+" = "+temp[1]);
}

字符串截取

从一个完整的字符串之中截取出部分内容。可用方法如下
在这里插入图片描述
代码示例: 观察字符串截取

String str = "helloworld" ; 
System.out.println(str.substring(5));
System.out.println(str.substring(0, 5));

注意事项:
1.索引从0开始
2. 注意前闭后开区间的写法, substring(0, 5) 表示包含 0 号下标的字符, 不包含 5 号下标

其他操作方法

在这里插入图片描述
代码示例: 观察trim()方法的使用

String str = "   hello world   " ; 
System.out.println("["+str+"]");
System.out.println("["+str.trim()+"]");

trim 会去掉字符串开头和结尾的空白字符(空格, 换行, 制表符等).
代码示例: 大小写转换

String str = "   hello%$$%@#$%world 哈哈哈 " ; 
System.out.println(str.toUpperCase());
System.out.println(str.toLowerCase());

这两个函数只转换字母。
代码示例: 观察isEmpty()方法

System.out.println("hello".isEmpty());
System.out.println("".isEmpty());
System.out.println(new String().isEmpty());

String类并没有提供首字母大写操作,需要自己实现.

代码示例: 首字母大写

public static void main(String[] args) {
 System.out.println(fistUpper("yuisama"));
 System.out.println(fistUpper(""));
 System.out.println(fistUpper("a"));
 }
 public static String fistUpper(String str) {
 if ("".equals(str)||str==null) {
 return str ; 
 }
 if (str.length()>1) {
 return str.substring(0, 1).toUpperCase()+str.substring(1) ; 
 }
 return str.toUpperCase() ;
 }

StringBuffer 和 StringBuilder

在String中使用"+"来进行字符串连接,但是这个操作在StringBuffer类中需要更改为append()方法:

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

范例:观察StringBuffer使用

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").append("www.bit.com.cn");
 }
}

String和StringBuffer最大的区别在于:String的内容无法修改,而StringBuffer的内容可以修改。频繁修改字符串的情况考虑使用StingBuffer。
注意:
1.String和StringBuffer类不能直接转换。如果要想互相转换,可以采用如下原则:
2.String变为StringBuffer:利用StringBuffer的构造方法或append()方法
3.StringBuffer变为String:调用toString()方法。
4.除了append()方法外,StringBuffer也有一些String类没有的方法
5.字符串反转

public synchronized StringBuffer reverse()

代码示例: 字符串反转

StringBuffer sb = new StringBuffer("helloworld");
System.out.println(sb.reverse());

删除指定范围的数据:

public synchronized StringBuffer delete(int start, int end)

代码示例: 观察删除操作

StringBuffer sb = new StringBuffer("helloworld");
System.out.println(sb.delete(5, 10));

插入数据

public synchronized StringBuffer insert(int offset, 各种数据类型 b)

代码示例: 观察插入操作

StringBuffer sb = new StringBuffer("helloworld");
System.out.println(sb.delete(5, 10).insert(0, "你好"));

面试题:请解释String、StringBuffer、StringBuilder的区别:
1.String的内容不可修改,StringBuffer与StringBuilder的内容可以修改.
2.StringBuffer与StringBuilder大部分功能是相似的
3.StringBuffer采用同步处理,属于线程安全操作;而StringBuilder未采用同步处理,属于线程不安全操作
在这里插入图片描述
在这里插入图片描述

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值