Java学习DAY9~认识String类(一)~2021.01.18

本文深入探讨了Java中字符串的创建,包括直接赋值、使用构造方法以及通过字符数组的方式。强调了字符串比较时`==`与`equals()`的区别,指出`equals()`是用于内容比较,而`==`比较的是对象引用。讲解了字符串常量池的概念和`intern()`方法的作用。此外,还详细解释了字符串的不可变性及其带来的好处。最后,讨论了字符、字节与字符串之间的转换,并提供了相关代码示例。
摘要由CSDN通过智能技术生成

1. 创建字符串

常见的构造 String 的方式

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

注意事项:
“hello” 这样的字符串字面值常量, 类型也是 String.
String 也是引用类型.

2. 字符串比较相等

  • String 使用 “==”比较并不是在比较字符串内容, 而是比较两个引用是否是指向同一个对象.
  • 关于对象的比较:面向对象编程语言中, 涉及到对象的比较, 有三种不同的方式, 比较身份, 比较值,较类型.
  • 在大部分编程语言中 “==”是用来比较比较值的. 但是 Java 中是用来比较身份的.
  • 如何理解比较值和比较身份呢?
    可以想象一个场景, 现在取快递, 都有包裹储物柜. 上面有很多的格子. 每个格子里面都放着东西.如下图所示:
    在这里插入图片描述

如上图所示"第二行, 左数第五列" 这个柜子和 “第二行, 右数第二列” 这个柜子是同一个柜子, 就是 身份相同. 如果身份相同, 那么里面放的东西一定也相同 (值一定也相同).例如, “第一行, 左数第一列” 这个柜子和 “第一行, 左数第二列” 这两个柜子不是同一个柜子, 但是柜子打开发现里面放着两双一模一样的鞋子,这就是“值相同”。

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

String str1 = new String("Hello");
String str2 = new String("Hello");
System.out.println(str1.equals(str2));

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

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类对象,此时对象池之中如若有指定内容,将直接进行引用
  • 如若没有,则开辟新的字符串对象而后将其保存在对象池之中以供下次使用
    理解 “池” (pool):
    “池” 是编程中的一种常见的, 重要的提升效率的方式, 我们会在未来的学习中遇到各种 “内存池”,“线程池”, “数据库连接池” …

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

综上, 我们一般采取直接赋值的方式创建 String 对象.

4. 理解字符串不可变

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

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

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

在这里插入图片描述
+= 之后 str 打印的结果却是变了, 但是不是 String 对象本身发生改变, 而是 str 引用到了其他的对象.
那么如果实在需要修改字符串,
a) 常见办法: 借助原字符串, 创建新的字符串

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

b) 特殊办法(选学): 使用 “反射” 这样的操作可以破坏封装, 访问一个类内部的 private 成员.
IDEA 中 ctrl + 左键 跳转到 String 类的定义, 可以看到内部包含了一个 char[] , 保存了字符串的内容.
在这里插入图片描述

String str = "Hello";
// 获取 String 类中的 value 字段. 这个 value 和 String 源码中的 value 是匹配的.
Field valueField = String.class.getDeclaredField("value");
// 将这个字段的访问属性设为 true
valueField.setAccessible(true);
// 把 str 中的 value 属性获取到.
char[] value = (char[]) valueField.get(str);
// 修改 value 的值
value[0] = 'h';
System.out.println(str);
// 执行结果
hello

关于反射:
反射是面向对象编程的一种重要特性, 有些编程语言也称为 “自省”.
指的是程序运行过程中, 获取/修改某个对象的详细信息(类型信息, 属性信息等), 相当于让一个对象更好的 “认清自己” .
为什么 String 要不可变?(不可变对象的好处是什么?):

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

5. 字符, 字节与字符串

5.1 字符与字符串

字符串内部包含一个字符数组,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’"之间的内容,如果是则为数字.

package javaSE_0118_faceObject;

public class IsNumbers {
    public static void main(String[] args) {
        String str = "1,a,2,3,4,5,6,7,8,9";
        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;
    }
}

5.2 字节与字符串

在这里插入图片描述

字节常用于数据传输以及编码转换的处理之中,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));

5.3 小结

何时使用 byte[], 何时使用 char[]:

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

常见面试题

面试题:请解释String类中两种对象实例化的区别

  1. 直接赋值:只会开辟一块堆内存空间,并且该字符串对象可以自动保存在对象池中以供下次
    使用。
  2. 构造方法:会开辟两块堆内存空间,不会自动保存在对象池中,可以使用intern()方法手工入
    池。
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值