String 字符串对象

String是什么

String字符串,是一种引用数据类型,并不是基础数据类型。

对于基础数据类型和引用数据类型的区别:

基础数据类型,在创建时直接将值存放在栈内存中。

引用数据类型,在创建时栈内存中存放一个引用,这个引用存放的是堆内存的位置,而堆内存中就是存放具体的值。

举例说明:

假如String对象是一个储物柜,在使用储物柜时(相当于新建一个String对象),

我们需要会得到一张记着储物柜的小票(小票相当于栈内存空间,小票上保存引用),

凭借这张小票就可以找到储物柜,然后拿到储物柜中存放的东西(储物柜相当于堆内存空间,可以根据引用去得到堆内存中的值)。

对于基础数据类型,则是直接将值写在小票上。

String的底层数据结构

String的底层实现是字符数组,这一点可以在源码中看到,

其中属性 private final char value[]; 就是用来存储String的值,

以下是String类的部分源码:

 1 public final class String
 2     implements java.io.Serializable, Comparable<String>, CharSequence {
 3     /** The value is used for character storage. */
 4     private final char value[];
 5 
 6     /** Cache the hash code for the string */
 7     private int hash; // Default to 0
 8 
 9     /** use serialVersionUID from JDK 1.0.2 for interoperability */
10     private static final long serialVersionUID = -6849794470754667710L;

String类被final修饰,则表示String类不可以被继承。

String类的value属性使用了 final 进行修饰,则表示value是一个常量,常量不可以更改。

String对象的重新赋值

String类的源码中可以看到String类的value属性使用了 final 进行修饰,既然是常量。

在发生对象需要重新赋值的时候,String的做法是,将String对象的栈内存与堆内存的引用断开。

再将栈内存中的引用指向新的堆内存空间。

本质上就是新建一个String对象。

由此可以引发一个问题,如果代码中频繁出现对String对象的重新赋值意味着会有大量的堆内存被弃用。

这部分被弃用的堆内存空间只能通过垃圾回收机制进行回收,这会降低内存的利用率。(浪费内存是一方面,另一方面垃圾回收也会耗费资源。此问题在面试中出现的概率比较高)

这个问题可以通过使用StringBuffer或者StringBuilder来解决(StringBuffer和StringBuilder会在接下来的文章进行介绍)。

以下是《Java开发实战经典》一书中的示例:

 

String对象的比较

比较方式分为两种(自定义比较方法除外):

a.双等号(==):双等号的判断依据是对象的堆内存地址是否相同。(用上面储物柜的例子就是同一个储物柜,当然放的是同样的东西)

b.使用方法equals(Object anObject):方法比较的是对象的取值。

String类中equals(Object anObject)方法源码:

 1     public boolean equals(Object anObject) {
 2         if (this == anObject) {
 3             return true;
 4         }
 5         if (anObject instanceof String) {
 6             String anotherString = (String)anObject;
 7             int n = value.length;
 8             if (n == anotherString.value.length) {
 9                 char v1[] = value;
10                 char v2[] = anotherString.value;
11                 int i = 0;
12                 while (n-- != 0) {               //通过while循环比较字符数组中的值
13                     if (v1[i] != v2[i])
14                         return false;
15                     i++;
16                 }
17                 return true;
18             }
19         }
20         return false;
21     }

 

以下是代码示例:

public class TestString {
    public static void main(String[] args) {
        String str1 = "a";
        String str2 = "a";
        String str3 = new String("a");   //使用new关键词创建String对象
        String str4 = "b";
        
        System.out.println("str1  ==    str2: "+(str1 == str2));
        System.out.println("str1 equals str2: "+(str1.equals(str2)));
        System.out.println("str1  ==    str3: "+(str1 == str3));
        System.out.println("str1 equals str3: "+(str1.equals(str3)));
        System.out.println("str1  ==    str4: "+(str1 == str4));
    }
}

 

控制台结果:

str1  ==    str2: true
str1 equals str2: true
str1  ==    str3: false
str1 equals str3: true
str1  ==    str4: false

通过 str1 == str2: true 结果得出的结论:

String类在不使用new 关键词来新建对象时,如果str1 和 str2 的取值相同,

并不会为str1 和 str2 分别开辟堆内存空间,而是将str1 和 str2同时指向同一个堆内存,减少内存浪费。

String类常用方法

以下是《Java开发实战经典》一书中表格:

 

 

结语

纸上得来终觉浅,绝知此事要躬行。

String类中的内容很多,单构造方法就是十多种。

虽然内容多,但是难度都不高可能花看一个小视频的时间能看完。

这里对String类进行简单的介绍。

 

posted @ 2019-04-29 12:31  超级珍贵  阅读(3701)  评论(0编辑  收藏  举报