Official account: Java Xiaokaxiu, website: Javaxks.com
Author: d, often link: blog.csdn.net/o9109003234/article/details/109523691
Is String immutable?
public class App { public static void main(String[] args) { String a = "111"; a = "222"; System.out.println(a); }}Copy the code
Some people would think the above code would print: 111
So that fits the invariance up here.
Hahaha, but not like that.
222
It’s not right, isn’t it? How did it change?
In fact, when the JVM runs, strings are given a separate piece of land.
The above:
Stirng a = "111";Copy the code
We know that string allocation, like any other object allocation, is expensive in time and space, and we use a lot of strings. The JVM implements some optimizations when instantiating strings to improve performance and reduce memory overhead:
Use a string constant pool. Whenever we create a string constant, the JVM first checks the string constant pool and returns a reference to the instance in the constant pool if the string already exists there. If the string does not exist in the constant pool, it is instantiated and placed in the constant pool. Because of the immutability of String strings, we can be pretty sure that there can never be two identical strings in the constant pool.
If you find the constant pool in JVM, you don’t need to create the object. Instead, you assign the reference address of the object to A. Unable to find it, create a new object and assign its reference address to A. In the same way a = “222”; If you can’t find it, create a new object, and then assign the reference address of the object to A.
Search the public vertical number: MarkerHub, follow the reply [vue] to get the beginning of the front and back end tutorial!
Did you notice the “reference address” in my description above? Object obj = new Object(); Obj is not an Object. It is just a variable that holds a reference address to an Object.
A variable declared by a reference type is one that actually stores a reference address in memory and an entity in the heap.
So many articles on the Internet like to say that
User user = new User()
Copy the code
Creates a user object, which I like to call an object. No rebuttal will be accepted here.
So String a = “111”; A contains a reference to the object “111”. The variable can be changed. The variable that cannot be changed is “111”.
Why is String immutable?
Simply put, the String class uses final keyword character arrays to hold strings. The code is as follows:
public final class String implements java.io.Serializable, Comparable<String>, CharSequence { private final char value[]; public String() { this.value = "".value; } public String(String original) { this.value = original.value; this.hash = original.hash; } public String(char value[]) { this.value = Arrays.copyOf(value, value.length); }}Copy the code
Three points can be seen from the above source code:
The String class is final
String stores content using a char array
Char arrays are final
I have to review it here. What’s the use of final?
When you modify a class with final, it indicates that the class cannot be inherited. That is, if a class is never inherited, it can be modified with final. Member variables ina final class can be set to final if necessary, but be aware that all member methods ina final class are implicitly specified as final methods.
When a final modifier means that the method is final, it cannot be overridden (multiple final modifier methods can be overridden). One thing to note here: The premise of rewriting is that the subclass can inherit the method from the parent class. If the access control permission of the final modified method in the parent class is private, the subclass cannot inherit the method directly. Therefore, the same method name and parameters can be defined in the subclass at this time, and the contradiction between rewriting and final will no longer occur. Instead, new methods are redefined in subclasses. (Note: Private methods of a class are implicitly specified as final methods.)
When final modifies a base datatype, it means that the value of the base datatype cannot change once initialized. When final modifies a reference type, it cannot be made to point to any other object after it is initialized, but the contents of the object to which the reference refers can change. It’s essentially the same thing, because the referenced value is an address, and final requires that the value, the value of the address, not change. In addition, final modifies a member variable (property) and must be displayed initialized. There are two ways to initialize it: (1. Assign it at declaration time, otherwise it must be assigned in all constructors of its class)
Such as:
/** * public class FinalDemo {private final String name; public FinalDemo(String name) { this.name = name; } public FinalDemo() { } }Copy the code
This is an error
So much for final
Let’s look at an example of using String
/** * Description: ** @author: 2020/11/3 * Welcome to our official account: Public class StringDemo {public static void main(String[] args) {String name = "old "; name.concat("!" ); System.out.println(name); System.out.println(name.concat("!" )); }}Copy the code
The output
Along the way, several common methods in String source code
public String concat(String str) { int otherLen = str.length(); If (otherLen == 0) {return this; if (otherLen == 0) {return this; } int len = value.length; char buf[] = Arrays.copyOf(value, len + otherLen); str.getChars(buf, len); // See? Return new String(buf, true); } void getChars(char dst[], int dstBegin) { System.arraycopy(value, 0, dst, dstBegin, value.length); } public String replace(char oldChar, char newChar) {// If the two are the same, it is necessary to replace. = newChar) { int len = value.length; int i = -1; Char [] val = value; // Copy the current char array to val. while (++i < len) { if (val[i] == oldChar) { break; }} if (I < len) {// Create a new char array char buf[] = new char[len]; for (int j = 0; j < i; j++) { buf[j] = val[j]; } while (i < len) { char c = val[i]; buf[i] = (c == oldChar) ? newChar : c; i++; } // create a new String object return new String(buf, true); } } return this; } public String substring(int beginIndex, int endIndex) { if (beginIndex < 0) { throw new StringIndexOutOfBoundsException(beginIndex); } if (endIndex > value.length) { throw new StringIndexOutOfBoundsException(endIndex); } int subLen = endIndex - beginIndex; if (subLen < 0) { throw new StringIndexOutOfBoundsException(subLen); Return ((beginIndex == 0) && (endIndex == value.length))? this : new String(value, beginIndex, subLen); } public String trim() { int len = value.length; int st = 0; char[] val = value; /* avoid getfield opcode */ while ((st < len) && (val[st] <= ' ')) { st++; } while ((st < len) && (val[len - 1] <= ' ')) { len--; } // If the string contains Spaces, call substring, otherwise nothing is done. Then, a new String object to return to the return ((st > 0) | | (len < value. Length)? substring(st, len) : this; }Copy the code
The concat, replace, SubString, or trim methods do not operate on the original string, but regenerate a new string object. That is, the original string is not changed.
Two conclusions are drawn:
Once a String is created, it is immutable. Any changes to the String do not affect the original object, and any related changes to the operation create a new object.
A new String is created every time a String is changed.
Go back to the previous example
//String a = "111"; Char data [] ={'1','1','1'}; Stirng a = new String(data); //a = "222"; char data [] ={'2','2','2'}; a = new String(data);Copy the code
This variable a holds a reference to the String for “222”.
Continue with the code below
public class App { public static void main(String[] args) { String a = "111"; String a1 = "111"; String b = new String("111"); // The object address is the same system.out.println (a==a1); // The object content is the same system.out.println (a.equals(a1)); System.out.println(a==b); System.out.println(a.equals(b)); }}Copy the code
The output
true
true
false
true
Copy the code
The first output is true, indicating that both variables a and A1 hold the same reference address.
The second also prints true, indicating that both a and A1 refer to the same address.
A and A1 are placed on the stack, holding the reference addresses of objects.
The object of new is in the heap.
Constants actually depend on the JDK version.
So String a = “111”; In the JVM request memory store “111” corresponding object, and save the object. When the String a1 = “1111”; “, will first go to the JVM’s field to see if there is a “111”, just before the save, so find, and then directly to the object reference address a1. So both A and A1 are holding the same reference address.
Anyone who touches Java knows that you can new an object. So String b = new String(“111″); You create an object and you assign the object reference address to variable B. But there is a special point here, and that is (” 111 “), which will first look in the JVM field and find the direct reference address. Could not find the argument constructor that creates an object and then references the address to a String.
So false is printed in the third one, because object references held by A and B are different.
The last output is true. That’s because both variables hold the contents of the reference address as “111”.
The answer:
If the constant pool exists, only one object needs to be created, otherwise two objects need to be created.