String的认识

创建String类

常见的构造String的方法:

//方法一
String str="Hello";
//方法二
String str =new String ("Hello");
//方法三
char[] ch={'H','e'.'l','l','o'};
String str=new String (ch);

注意事项:

“hello” 这样的字符串字面值常量, 类型也是 String.

String 也是引用类型. String str = “Hello”; 这样的代码内存布局如下

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-lQKeymJc-1638691291122)(C:/Users/78581/AppData/Roaming/Typora/typora-user-images/image-20211121172912508.png)]

由于 String 是引用类型, 因此对于以下代码

String str1 = "Hello";

String str2 = str1;

内存布局如图

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-i2rMfVoK-1638691291123)(C:/Users/78581/AppData/Roaming/Typora/typora-user-images/image-20211122121452707.png)]

那么有同学可能会说, 是不是修改 str1 , str2 也会随之变化呢?

str1 = "world";

System.out.println(str2);

// 执行结果

Hello

我们发现, “修改” str1 之后, str2 也没发生变化, 还是 hello?

事实上, str1 = “world” 这样的代码并不算 “修改” 字符串, 而是让 str1 这个引用指向了一个新的 String 对象.

字符串比较相等

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

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

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

代码1

String str1 = "Hello";

String str2 = "Hello"; 

System.out.println(str1 == str2); 

// 执行结果

true 

看起来貌似没啥问题, 再换个代码试试, 发现情况不太妙.

代码2

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

我们来分析两种创建 String 方式的差异.

代码1内存布局

image-20211122121452707

我们发现, str1 和 str2 是指向同一个对象的. 此时如 “Hello” 这样的字符串常量是在 字符串常量池 中.

关于字符串常量池

如 “Hello” 这样的字符串字面值常量, 也是需要一定的内存空间来存储的. 这样的常量具有一个特点, 就是不需要

修改(常量嘛). 所以如果代码中有多个地方引用都需要使用 “Hello” 的话, 就直接引用到常量池的这个位置就行

了, 而没必要把 “Hello” 在内存中存储两次.

代码2内存布局

image-20211122122049827

通过 String str1 = new String(“Hello”); 这样的方式创建的 String 对象相当于再堆上另外开辟了空间来存储

“Hello” 的内容, 也就是内存中存在两份 “Hello”.

String 使用 == 比较并不是在比较字符串内容**,** 而是比较两个引用是否是指向同一个对象.

关于对象的比较

面向对象编程语言中, 涉及到对象的比较, 有三种不同的方式, 比较身份, 比较值, 比较类型.

在大部分编程语言中 == 是用来比较比较值的. 但是 Java 中的 == 是用来比较身份的.

如何理解比较值和比较身份呢?

可以想象一个场景, 现在取快递, 都有包裹储物柜. 上面有很多的格子. 每个格子里面都放着东西.

例如, “第二行, 左数第五列” 这个柜子和 “第二行, 右数第二列” 这个柜子是同一个柜子, 就是 身份相同. 如果身份

相同, 那么里面放的东西一定也相同 (值一定也相同).

例如, “第一行, 左数第一列” 这个柜子和 “第一行, 左数第二列” 这两个柜子不是同一个柜子, 但是柜子打开后发现

里面放着的是完全一模一样的两双鞋子. 这个时候就是 值相同

Java 中要想比较字符串的内容, 必须采用String类提供的equals方法.

equals 使用注意事项

现在需要比较 str 和 “Hello” 两个字符串是否相等, 我们该如何来写呢?

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 str = null;
// 方式一
System.out.println(str.equals("Hello"));  // 执行结果 抛出 java.lang.NullPointerException 异 常
// 方式二
System.out.println("Hello".equals(str));	//执行结果 false

注意事项: “Hello” 这样的字面值常量, 本质上也是一个 String 对象, 完全可以使用 equals 等 String 对象的方法

因为引用类型变量存储的是地址,所以“=”比较的是两个引用变量的地址是否相同,而不是比较内容

String重写的equals方法比较的就是变量中的每一个具体值

字符串常量池

在上面的例子中, String类的两种实例化操作, 直接赋值和 new 一个新的 String

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底层实际上会自动维护一个对象池(字符串常量池)

  1. 如果现在采用了直接赋值的模式进行String类的对象实例化操作,那么该实例化对象(字符串内容)将自动保存

到这个对象池之中.

  1. 如果下次继续使用直接赋值的模式声明String类对象,此时对象池之中如若有指定内容,将直接进行引用

  2. 如若没有,则开辟新的字符串对象而后将其保存在对象池之中以供下次使用

理解 ” (pool)

“池” 是编程中的一种常见的, 重要的提升效率的方式, 我们会在未来的学习中遇到各种 “内存池”, “线程池”, "数据

库连接池" …

然而池这样的概念不是计算机独有, 也是来自于生活中.

b) 采用构造方法

类对象使用构造方法实例化是标准做法。分析如下程序:

String str = new String("hello") ;

image-20211122123306272

这样的做法有两个缺点:

  1. 如果使用String构造方法就会开辟两块堆内存空间,并且其中一块堆内存将成为垃圾空间(字符串常量 “hello” 也

是一个匿名对象, 用了一次之后就不再使用了, 就成为垃圾空间, 会被 JVM 自动回收掉).

  1. 字符串共享问题. 同一个字符串可能会被存储多次, 比较浪费空间

我们可以使用 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 对象

理解字符串的不可变

字符串是一种不可变对象. 它的内容不可改变.

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

感受下形如这样的代码

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

形如 += 这样的操作, 表面上好像是修改了字符串, 其实不是. 内存变化如下:

image-20211122124550762

字符、字节与字符串

字符和字符串

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

代码示例: 获取指定位置的字符

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[] 相互转换

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

代码示例:实现字符串与字节数组的转换处理

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[] 呢?

  1. byte[] 是把 String 按照一个字节一个字节的方式处理, 这种适合在网络传输, 数据存储这样的场景下使用. 更适合

针对二进制数据来操作.

  1. char[] 是吧 String 按照一个字符一个字符的方式处理, 更适合针对文本数据来操作, 尤其是包含中文的时候.

回忆概念: 文本数据 vs 二进制数据

一个简单粗暴的区分方式就是用记事本打开能不能看懂里面的内容.

如果看的懂, 就是文本数据(例如 .java 文件), 如果看不懂, 就是二进制数据(例如.class文件)

字符串常见操作

上面使用过String类提供的equals()方法,该方法本身是可以进行区分大小写的相等判断。除了这个方法之外,String

类还提供有如下的比较操作:

字符串比较

No方法名称类型描述
1public boolean equals(Object anObject)普通区分大小写的比较
2public boolean equalsIanore(Object anObject)普通不区分大小写的比较
3public int conpareTo(String anotherString)普通比较两个字符串大小关系

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

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

字符串的比较大小规则, 总结成三个字 “字典序” 相当于判定两个字符串在一本词典的前面还是后面. 先比较第一

个字符的大小(根据 unicode 的值来判定), 如果不分胜负, 就依次比较后面的内容

字符串查找

从一个完整的字符串之中可以判断指定内容是否存在,对于查找方法有如下定义:

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

代码示例**😗* 字符串查找,最好用最方便的就是contains()

String str = "helloworld" ; 

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

该判断形式是从JDK1.5之后开始追加的,在JDK1.5以前要想实现与之类似的功能,就必须借助、indexOf()方法完

成。

代码示例: 使用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("可以查到指定字符串!"); 

} 

现在基本都是用contains()方法完成。

使用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

字符串替换

使用一个指定的新的字符串替换掉已有的字符串数据,可用的方法如下

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

代码示例: 字符串的替换处理

String str = "helloworld" ; 

System.out.println(str.replaceAll("l", "_")); 

System.out.println(str.replaceFirst("l", "_")); 

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

字符串拆分

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

代码示例: 实现字符串的拆分处理

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]); 
} 

这种代码在以后的开发之中会经常出现

字符串截取

从一个完整的字符串之中截取出部分内容。可用方法如下:

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

代码示例**😗* 观察字符串截取

String str = "helloworld" ; 

System.out.println(str.substring(5)); 

System.out.println(str.substring(0, 5)); 

注意事项**😗*

  1. 索引从0开始

  2. 注意前闭后开区间的写法, substring(0, 5) 表示包含 0 号下标的字符, 不包含 5 号下标

其他操作方法

No方法名称类型描述
1public String trim( )普通去除字符串中的左右空格,保留中间的空格
2public String toUpperCase( )普通字符串大写
3public String toLowerCase( )普通字符串小写
4public native String intern( )普通字符串入池操作
5public String concat(String str )普通字符串连接,等同于“ + ”,不入池
6public int length( )普通取得字符串长度
7public boolean isEmpty( )普通判断是否为空字符串,但不是null,而是长度0

代码示例: 观察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()); 

这两个函数只转换字母。

代码示例**😗* 字符串length()

String str = " hello%$$%@#$%world 哈哈哈 " ; 

System.out.println(str.length()); 

注意:数组长度使用数组名称.length属性,而String中使用的是length()方法

代码示例: 观察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类的特点:

任何的字符串常量都是String对象,而且String的常量一旦声明不可改变,如果改变对象内容,改变的是其引用的指

向而已。

通常来讲String的操作比较简单,但是由于String的不可更改特性,为了方便字符串的修改,提供StringBuffffer和

StringBuilder类。

StringBuffffer 和 StringBuilder 大部分功能是相同的,我们课件上主要介绍 StringBuffffer

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

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

范例:观察StringBuffffer使用

 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("hhh"); 
	 } 
 } 

String和StringBuffffer最大的区别在于:String的内容无法修改,而StringBuffffer的内容可以修改。频繁修改字符串的

情况考虑使用StingBuffffer。

为了更好理解String和StringBuffffer,我们来看这两个类的继承结构:

String类StringBuffer类
public fifinal class String implementspublic fifinal class StringBuffffer extends

可以发现两个类都是"CharSequence"接口的子类。这个接口描述的是一系列的字符集。所以字符串是字符集的子

类,如果以后看见CharSequence,最简单的联想就是字符串。

注意:String和StringBuffffer类不能直接转换。如果要想互相转换,可以采用如下原则:

String变为StringBuffffer:利用StringBuffffer的构造方法或append()方法

StringBuffffer变为String:调用toString()方法。

除了append()方法外,StringBuffffer也有一些String类没有的方法:

字符串反转:

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、StringBuffffer、StringBuilder的区别:

String的内容不可修改,StringBuffffer与StringBuilder的内容可以修改.

StringBuffffer与StringBuilder大部分功能是相似的

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值