auto commit
This commit is contained in:
parent
dce6d0ee3f
commit
322a6c58bf
|
@ -6,7 +6,8 @@
|
||||||
* [概览](#概览)
|
* [概览](#概览)
|
||||||
* [不可变的好处](#不可变的好处)
|
* [不可变的好处](#不可变的好处)
|
||||||
* [String, StringBuffer and StringBuilder](#string,-stringbuffer-and-stringbuilder)
|
* [String, StringBuffer and StringBuilder](#string,-stringbuffer-and-stringbuilder)
|
||||||
* [String.intern()](#stringintern)
|
* [String Pool](#string-pool)
|
||||||
|
* [new String("abc")](#new-string"abc")
|
||||||
* [三、运算](#三运算)
|
* [三、运算](#三运算)
|
||||||
* [参数传递](#参数传递)
|
* [参数传递](#参数传递)
|
||||||
* [float 与 double](#float-与-double)
|
* [float 与 double](#float-与-double)
|
||||||
|
@ -64,7 +65,7 @@ int y = x; // 拆箱
|
||||||
|
|
||||||
new Integer(123) 与 Integer.valueOf(123) 的区别在于:
|
new Integer(123) 与 Integer.valueOf(123) 的区别在于:
|
||||||
|
|
||||||
- new Integer(123) 每次都会新建一个对象
|
- new Integer(123) 每次都会新建一个对象;
|
||||||
- Integer.valueOf(123) 会使用缓存池中的对象,多次调用会取得同一个对象的引用。
|
- Integer.valueOf(123) 会使用缓存池中的对象,多次调用会取得同一个对象的引用。
|
||||||
|
|
||||||
```java
|
```java
|
||||||
|
@ -193,33 +194,90 @@ String 不可变性天生具备线程安全,可以在多个线程中安全地
|
||||||
|
|
||||||
[StackOverflow : String, StringBuffer, and StringBuilder](https://stackoverflow.com/questions/2971315/string-stringbuffer-and-stringbuilder)
|
[StackOverflow : String, StringBuffer, and StringBuilder](https://stackoverflow.com/questions/2971315/string-stringbuffer-and-stringbuilder)
|
||||||
|
|
||||||
## String.intern()
|
## String Pool
|
||||||
|
|
||||||
使用 String.intern() 可以保证相同内容的字符串变量引用同一的内存对象。
|
字符串常量池(String Poll)保存着所有字符串字面量(literal strings),这些字面量在编译时期就确定。不仅如此,还可以使用 String 的 intern() 方法在运行过程中将字符串添加到 String Poll 中。
|
||||||
|
|
||||||
下面示例中,s1 和 s2 采用 new String() 的方式新建了两个不同对象,而 s3 是通过 s1.intern() 方法取得一个对象引用。intern() 首先把 s1 引用的对象放到 String Pool(字符串常量池)中,然后返回这个对象引用。因此 s3 和 s1 引用的是同一个字符串常量池的对象。
|
当一个字符串调用 intern() 方法时,如果 String Poll 中已经存在一个字符串和该字符串值相等(使用 equals() 方法进行确定),那么就会返回 String Poll 中字符串的引用;否则,就会在 String Poll 中添加一个新的字符串,并返回这个新字符串的引用。
|
||||||
|
|
||||||
|
下面示例中,s1 和 s2 采用 new String() 的方式新建了两个不同字符串,而 s3 和 s4 是通过 s1.intern() 方法取得一个字符串引用。intern() 首先把 s1 引用的字符串放到 String Pool 中,然后返回这个字符串引用。因此 s3 和 s4 引用的是同一个字符串。
|
||||||
|
|
||||||
```java
|
```java
|
||||||
String s1 = new String("aaa");
|
String s1 = new String("aaa");
|
||||||
String s2 = new String("aaa");
|
String s2 = new String("aaa");
|
||||||
System.out.println(s1 == s2); // false
|
System.out.println(s1 == s2); // false
|
||||||
String s3 = s1.intern();
|
String s3 = s1.intern();
|
||||||
System.out.println(s1.intern() == s3); // true
|
String s4 = s1.intern();
|
||||||
|
System.out.println(s3 == s4); // true
|
||||||
```
|
```
|
||||||
|
|
||||||
如果是采用 "bbb" 这种使用双引号的形式创建字符串实例,会自动地将新建的对象放入 String Pool 中。
|
如果是采用 "bbb" 这种字面量的形式创建字符串,会自动地将字符串放入 String Pool 中。
|
||||||
|
|
||||||
```java
|
```java
|
||||||
String s4 = "bbb";
|
|
||||||
String s5 = "bbb";
|
String s5 = "bbb";
|
||||||
|
String s6 = "bbb";
|
||||||
System.out.println(s4 == s5); // true
|
System.out.println(s4 == s5); // true
|
||||||
```
|
```
|
||||||
|
|
||||||
在 Java 7 之前,字符串常量池被放在运行时常量池中,它属于永久代。而在 Java 7,字符串常量池被移到堆中。这是因为永久代的空间有限,在大量使用字符串的场景下会导致 OutOfMemoryError 错误。
|
在 Java 7 之前,String Poll 被放在运行时常量池中,它属于永久代。而在 Java 7,String Poll 被移到堆中。这是因为永久代的空间有限,在大量使用字符串的场景下会导致 OutOfMemoryError 错误。
|
||||||
|
|
||||||
- [StackOverflow : What is String interning?](https://stackoverflow.com/questions/10578984/what-is-string-interning)
|
- [StackOverflow : What is String interning?](https://stackoverflow.com/questions/10578984/what-is-string-interning)
|
||||||
- [深入解析 String#intern](https://tech.meituan.com/in_depth_understanding_string_intern.html)
|
- [深入解析 String#intern](https://tech.meituan.com/in_depth_understanding_string_intern.html)
|
||||||
|
|
||||||
|
## new String("abc")
|
||||||
|
|
||||||
|
使用这种方式一共会创建两个字符串对象(前提是 String Poll 中还没有 "abc" 字符串对象)。
|
||||||
|
|
||||||
|
- "abc" 属于字符串字面量,因此编译时期会在 String Poll 中创建一个字符串对象,指向这个 "abc" 字符串字面量;
|
||||||
|
- 而使用 new 的方式会在堆中创建一个字符串对象。
|
||||||
|
|
||||||
|
创建一个测试类,其 main 方法中使用这种方式来创建字符串对象。
|
||||||
|
|
||||||
|
```java
|
||||||
|
public class NewStringTest {
|
||||||
|
public static void main(String[] args) {
|
||||||
|
String s = new String("abc");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
使用 javap -verbose 进行反编译,得到以下内容:
|
||||||
|
|
||||||
|
```java
|
||||||
|
// ...
|
||||||
|
Constant pool:
|
||||||
|
// ...
|
||||||
|
#2 = Class #18 // java/lang/String
|
||||||
|
#3 = String #19 // abc
|
||||||
|
// ...
|
||||||
|
#18 = Utf8 java/lang/String
|
||||||
|
#19 = Utf8 abc
|
||||||
|
// ...
|
||||||
|
|
||||||
|
public static void main(java.lang.String[]);
|
||||||
|
descriptor: ([Ljava/lang/String;)V
|
||||||
|
flags: ACC_PUBLIC, ACC_STATIC
|
||||||
|
Code:
|
||||||
|
stack=3, locals=2, args_size=1
|
||||||
|
0: new #2 // class java/lang/String
|
||||||
|
3: dup
|
||||||
|
4: ldc #3 // String abc
|
||||||
|
6: invokespecial #4 // Method java/lang/String."<init>":(Ljava/lang/String;)V
|
||||||
|
9: astore_1
|
||||||
|
// ...
|
||||||
|
```
|
||||||
|
|
||||||
|
在 Constant Poll 中,#19 存储这字符串字面量 "abc",#3 是 String Poll 的字符串对象,它指向 #19 这个字符串字面量。在 main 方法中,0: 行使用 new #2 在堆中创建一个字符串对象,并且使用 ldc #3 将 String Poll 中的字符串对象作为 String 构造函数的参数。
|
||||||
|
|
||||||
|
以下是 String 构造函数的源码,可以看到,在将一个字符串对象作为另一个字符串对象的构造函数参数时,并不会完全复制 value 数组内容,而是都会指向同一个 value 数组。
|
||||||
|
|
||||||
|
```java
|
||||||
|
public String(String original) {
|
||||||
|
this.value = original.value;
|
||||||
|
this.hash = original.hash;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
# 三、运算
|
# 三、运算
|
||||||
|
|
||||||
## 参数传递
|
## 参数传递
|
||||||
|
|
Loading…
Reference in New Issue
Block a user