可以将文章内容翻译成中文,广告屏蔽插件可能会导致该功能失效(如失效,请关闭广告屏蔽插件后再试):
问题:
From when I learned that the class java.lang.String
is declared as final in Java, I was wondering why is that? I didn\'t find any answer back then, but this post: How to create a replica of String class in Java? reminded me of my query.
Sure, String provides all the functionality I ever needed, and never thought of any operation that would require an extension of class String, but still you\'ll never know what someone might need!
So, does anyone know what was the intent of the designers when they decided to make it final?
回答1:
It is very useful to have strings implemented as immutable objects. You should read about immutability to understand more about it.
One advantage of immutable objects is that
You can share duplicates by pointing them to a single instance.
(from here).
If String were not final, you could create a subclass and have two strings that look alike when \"seen as Strings\", but that are actually different.
回答2:
This is a nice article that outlines two reasons already mentioned on the above answers:
- Security: the system can hand out
sensitive bits of read-only
information without worrying that
they will be altered
- Performance: immutable data is very
useful in making things thread-safe.
And this probably is the most detailed comment in that article. Its has to do with the string pool in Java and security issues. Its about how to decide what goes into the string pool. Assuming both strings are equal if their sequence of characters are the same, then we have a race condition on who gets there first and along with it security issues. If not, then the string pool will contain redundant strings thus losing the advantage of having it in the first place. Just read it out for yourself, will ya?
Extending String would play havoc with equals and intern. JavaDoc says equals:
Compares this string to the specified object. The result is true if and only if the argument is not null and is a String object that represents the same sequence of characters as this object.
Assuming java.lang.String
wasn\'t final, a SafeString
could equal a String
, and vice versa; because they\'d represent the same sequence of characters.
What would happen if you applied intern
to a SafeString
-- would the SafeString
go into the JVM\'s string pool? The ClassLoader
and all objects the SafeString
held references to would then get locked in place for the lifetime of the JVM. You\'d get a race condition about who could be the first to intern a sequence of characters -- maybe your SafeString
would win, maybe a String
, or maybe a SafeString
loaded by a different classloader (thus a different class).
If you won the race into the pool, this would be a true singleton and people could access your whole environment (sandbox) through reflection and secretKey.intern().getClass().getClassLoader()
.
Or the JVM could block this hole by making sure that only concrete String objects (and no subclasses) were added to the pool.
If equals was implemented such that SafeString
!= String
then SafeString.intern
!= String.intern
, and SafeString
would have to be added to the pool. The pool would then become a pool of <Class, String>
instead of <String>
and all you\'d need to enter the pool would be a fresh classloader.
回答3:
The absolutely most important reason that String is immutable or final is that it is used by the class loading mechanism, and thus have profound and fundamental security aspects.
Had String been mutable or not final, a request to load \"java.io.Writer\" could have been changed to load \"mil.vogoon.DiskErasingWriter\"
reference : Why String is immutable in Java
回答4:
String
is a very core class in Java, many things rely on it working a certain way, for example being immutable.
Making the class final
prevents subclasses that could break these assumptions.
Note that, even now, if you use reflection, you can break Strings (change their value or hashcode). Reflection can be stopped with a security manager. If String
was not final
, everyone could do it.
Other classes that are not declared final
allow you to define somewhat broken subclasses (you could have a List
that adds to the wrong position, for example) but at least the JVM does not depend on those for its core operations.
回答5:
As Bruno said it\'s about immutability. It\'s not only about Strings but as well about any wrappers e.g. Double, Integer, Character, etc. There are many reasons for this:
- Thread safety
- Security
- Heap that is managed by Java itself (differently to ordinary heap that is Garbage Collected in different manner)
- Memory management
Basically it so you, as a programmer, can be sure that your string will never be changed. It as well, if you know how it works, can improve memory managemnt. Try to create two identical string one after another, for example \"hello\". You will notice, if you debug, that they have identical IDs, that means that they are exactly THE SAME objects. This is due to the fact that Java let\'s you do it. This wouldn\'t be posssible if the strings were muttable. They can have the same I\'d, etc., because they will never change. So if you ever decide to create 1,000,000 string \"hello\" what you\'d really do is create 1,000,000 pointers to \"hello\". As well alling any function on string, or any wrappers for that reason, would result in creating another object (again look at object ID - it will change).
Aditionally final in Java does not necessarily mean that object cannot change (it is different to for example C++). It means that the address to which it points cannot change, but you still can change it\'s properties and/or attributes. So understanding the difference between immutability and final in some case might be really important.
HTH
References:
- http://chaoticjava.com/posts/how-does-garbage-collection-work/
- http://chaoticjava.com/posts/gc-tips-and-memory-leaks/
- http://java.sun.com/j2se/1.5/pdf/jdk50_ts_guide.pdf
回答6:
It may have been to simplify implementation. If you design a class that will be inheritable by users of the class, then you have a whole new set of use cases to consider into your design. What happens if they do this or that with X proptected field? Making it final they can focus on getting the public interface working correctly and make sure it\'s solid.
回答7:
With a lot of good points already mentined I would like to add another one -one of the reason of Why String is immutable in Java is to allow String to cache its hashcode , being immutable String in Java caches its hashcode, and do not calculate every time we call hashcode method of String, which makes it very fast as hashmap key to be used in hashmap in Java.
In short because String is immutable, no one can change its contents once created which guarantees hashCode of String to be same on multiple invocation.
If you see String
class has is declared as
/** Cache the hash code for the string */
private int hash; // Default to 0
and hashcode()
function is as follows -
public int hashCode() {
int h = hash;
if (h == 0 && value.length > 0) {
char val[] = value;
for (int i = 0; i < value.length; i++) {
h = 31 * h + val[i];
}
hash = h;
}
return h;
}
If it is already computer just return the value.
回答8:
In addition to the reasons mentioned in other answers (security, immutability, performance) it should be noted that String
has special language support. You can write String
literals and there\'s support for the +
operator. Allowing programmers to subclass String
, would encourage hacks such as:
class MyComplex extends String { ... }
MyComplex a = new MyComplex(\"5+3i\");
MyComplex b = new MyComplex(\"7+4i\");
MyComplex c = new MyComplex(a + b); // would work since a and b are strings,
// and a string + a string is a string.
回答9:
Well, I have some different thought I am not sure whether I am correct or not but in Java String is the only object which can be treated as a primitive data type as well I mean we can create a String object as String name=\"java\". Now like others primitive datatypes which are copy by value not copy by reference String is expected to have same behavior so thats why String is final. Thats what my thought it. Please ignore if its completely illogical.
回答10:
The finality of strings also defends them as a standard. In C++ you can create subclasses of string, so every programming shop could have its own version of string. This would lead to a lack of a strong standard.
回答11:
To make sure we do not get a better implementation. It should of course have been an interface.
[edit] Ah, getting more clueless down votes. The answer is perfectly serious. I\'ve had to program my way around the stupid String implementation several times, leading to severe performance & productivity loss
回答12:
Apart from the obvious reasons suggested in other answers, one thought of making String class final could also be related to virtual methods performance overhead. Remember String is a heavy class, making this final, means no sub-implementation for sure, means no indirection calling overhead ever. Of course now we have things like virtual invoke and others, which always does these kind of optimization for you.
回答13:
Does JVM know what is immutable? Answer is No, The constant pool contains all immutable fields but all immutable fields/objects are not stored in constant pool only. Only we implement it in a way that it achieves immutablity and its features. CustomString could be implemented without making it final using MarkerInterface which would provide java special behaviour for its pooling, the feature is still awaited!
回答14:
Most of the answers are related to the immutability -- why an object of String type cannot be updated in place. There is a lot of good discussion here, and the Java community would do well to adopt immutability as a principal. (Not holding my breath.)
However the OP\'s question is about why it\'s final -- why can it not be extended. Some here did take this on, but I would agree with the OP that there is a real gap here. Other language allow devs create new nominal types for a type. For example in Haskell I can create the following new types that are identical at run-time as text, but provide bind-safety at compile time.
newtype AccountCode = AccountCode Text
newtype FundCode = FundCode Text
So I would put the following suggestion forward as an enhancement to the Java language:
newtype AccountCode of String;
newtype FundCode of String;
AccountCode acctCode = \"099876\";
FundCode fundCode = \"099876\";
acctCode.equals(fundCode); // evaluates to false;
acctCode.toString().equals(fundCode.toString()); // evaluates to true;
acctCode=fundCode; // compile error
getAccount(fundCode); // compile error
(Or perhaps we could start to ween ourselves off Java)
回答15:
Let\'s say you have an Employee
class that has a method greet
. When the greet
method is called it simply prints Hello everyone!
. So that is the expected behavior of greet
method
public class Employee {
void greet() {
System.out.println(\"Hello everyone!\");
}
}
Now, let GrumpyEmployee
subclass Employee
and override greet
method as shown below.
public class GrumpyEmployee extends Employee {
@Override
void greet() {
System.out.println(\"Get lost!\");
}
}
Now in the below code have a look at the sayHello
method. It takes Employee
instance as a parameter and calls the greet method hoping that it would say Hello everyone!
But what we get is Get lost!
. This change in behavior is because of Employee grumpyEmployee = new GrumpyEmployee();
public class TestFinal {
static Employee grumpyEmployee = new GrumpyEmployee();
public static void main(String[] args) {
TestFinal testFinal = new TestFinal();
testFinal.sayHello(grumpyEmployee);
}
private void sayHello(Employee employee) {
employee.greet(); //Here you would expect a warm greeting, but what you get is \"Get lost!\"
}
}
This situation can be avoided if the Employee
class was made final
. Now it\'s up to your imagination the amount of chaos a cheeky programmer could cause if String
Class was not declared as final
.
回答16:
If you create a string once It will consider that,it is a object if you want to modify that, it is not possible,it will create new object.