The hashCode value of a Java String is computed as (String.hashCode()):
s[0]*31^(n-1) + s[1]*31^(n-2) + ... + s[n-1]
Are there any circumstances (say JVM version, vendor, etc.) under which the following expression will evaluate to false?
boolean expression = "This is a Java string".hashCode() == 586653468
Update #1: If you claim that the answer is "yes, there are such circumstances" - then please give a concrete example of when "This is a Java string".hashCode() != 586653468. Try to be as specific/concrete as possible.
Update #2: We all know that relying on the implementation details of hashCode() is bad in general. However, I'm talking specifically about String.hashCode() - so please keep the answer focused to String.hashCode(). Object.hashCode() is totally irrelevant in the context of this question.
-
2Do you actually need this functionality ? Why do you need the precise value ?Brian Agnew– Brian Agnew2009年04月24日 10:01:39 +00:00Commented Apr 24, 2009 at 10:01
-
27@Brian: I'm trying to understand the contract of String.hashCode().knorv– knorv2009年04月24日 10:02:52 +00:00Commented Apr 24, 2009 at 10:02
-
3@Knorv Its not ncessary to understand exactly how it works - its more important to understand the contract and its ulterior meaning.mP.– mP.2009年04月24日 10:09:00 +00:00Commented Apr 24, 2009 at 10:09
-
47@mP: Thanks for your input, but I guess that's up to me to decide.knorv– knorv2009年04月24日 10:45:07 +00:00Commented Apr 24, 2009 at 10:45
-
why did they give the first character the largest power? when you want to optimize it for speed in order to preserve extra calculations, you would store the power of the previous one, yet the previous one would be from the last character to the first one. this mean there would also be cache misses. isn't it more efficient to have an algorithm of: s[0] + s[1]*31 + s[2]*31^2 + ... + s[n-1]*31^[n-1] ?android developer– android developer2013年11月18日 23:21:53 +00:00Commented Nov 18, 2013 at 23:21
8 Answers 8
I can see that documentation as far back as Java 1.2.
While it's true that in general you shouldn't rely on a hash code implementation remaining the same, it's now documented behaviour for java.lang.String
, so changing it would count as breaking existing contracts.
Wherever possible, you shouldn't rely on hash codes staying the same across versions etc - but in my mind java.lang.String
is a special case simply because the algorithm has been specified... so long as you're willing to abandon compatibility with releases before the algorithm was specified, of course.
-
9The documented behaviour of String has been specified since Java 1.2 In v1.1 of the API, the hash code computation is not specified for the String class.Martin OConnor– Martin OConnor2009年04月24日 11:34:07 +00:00Commented Apr 24, 2009 at 11:34
-
In this case we better write our own hashing codes 'ight matey?Felype– Felype2013年06月18日 10:39:50 +00:00Commented Jun 18, 2013 at 10:39
-
@Felype: I really don't know what you're trying to say here, I'm afraid.Jon Skeet– Jon Skeet2013年06月18日 11:30:47 +00:00Commented Jun 18, 2013 at 11:30
-
@JonSkeet I mean, in this case we may perhaps write our own code to generate our own hash, to grant portability. Is it?Felype– Felype2013年06月19日 16:16:49 +00:00Commented Jun 19, 2013 at 16:16
-
1most java code simply don't run in pre-1.4. it is safe to assume it is consistent.J-16 SDiZ– J-16 SDiZ2014年11月27日 10:50:15 +00:00Commented Nov 27, 2014 at 10:50
I found something about JDK 1.0 and 1.1 and>= 1.2:
In JDK 1.0.x and 1.1.x the hashCode function for long Strings worked by sampling every nth character. This pretty well guaranteed you would have many Strings hashing to the same value, thus slowing down Hashtable lookup. In JDK 1.2 the function has been improved to multiply the result so far by 31 then add the next character in sequence. This is a little slower, but is much better at avoiding collisions. Source: http://mindprod.com/jgloss/hashcode.html
Something different, because you seem to need a number: How about using CRC32 or MD5 instead of hashcode and you are good to go - no discussions and no worries at all...
You should not rely on a hash code being equal to a specific value. Just that it will return consistent results within the same execution. The API docs say the following :
The general contract of hashCode is:
- Whenever it is invoked on the same object more than once during an execution of a Java application, the hashCode method must consistently return the same integer, provided no information used in equals comparisons on the object is modified. This integer need not remain consistent from one execution of an application to another execution of the same application.
EDIT Since the javadoc for String.hashCode() specifies how a String's hash code is computed, any violation of this would violate the public API specification.
-
1Your answer is valid, but does not address the specific question asked.knorv– knorv2009年04月24日 09:18:47 +00:00Commented Apr 24, 2009 at 9:18
-
7That's the general hash code contract - but the specific contract for String gives details of the algorithm, and effectively overrides this general contract IMO.Jon Skeet– Jon Skeet2009年04月24日 09:33:19 +00:00Commented Apr 24, 2009 at 9:33
As said above, in general you should not rely on the hash code of a class remaining the same. Note that even subsequent runs of the same application on the same VM may produce different hash values. AFAIK the Sun JVM's hash function calculates the same hash on every run, but that's not guaranteed.
Note that this is not theoretical. The hash function for java.lang.String
was changed in JDK1.2 (the old hash had problems with hierarchical strings like URLs or file names, as it tended to produce the same hash for strings which only differed at the end).
java.lang.String
is a special case, as the algorithm of its hashCode()
is (now) documented, so you can probably rely on that. I'd still consider it bad practice. If you need a hash algorithm with special, documented properties, just write one :-).
-
5But was the algorithm specified in the docs before JDK 1.2? If not, it's a different situation. The algorithm is now laid down in the docs, so changing it would be a breaking change to a public contract.Jon Skeet– Jon Skeet2009年04月24日 09:27:02 +00:00Commented Apr 24, 2009 at 9:27
-
(I remember it as 1.1.) The original (poorer) algorithm was documented. Incorrectly. The documented algorithm actually threw an ArrayIndexOutOfBoundsException.Tom Hawtin - tackline– Tom Hawtin - tackline2009年04月24日 10:00:44 +00:00Commented Apr 24, 2009 at 10:00
-
@Jon Skeet: Ah, didn't know that the algorithm of String.hashCode() is documented. Of course that changes things. Updated my comment.sleske– sleske2009年04月27日 13:49:32 +00:00Commented Apr 27, 2009 at 13:49
Another (!) issue to worry about is the possible change of implementation between early/late versions of Java. I don't believe the implementation details are set in stone, and so potentially an upgrade to a future Java version could cause problems.
Bottom line is, I wouldn't rely on the implementation of hashCode()
.
Perhaps you can highlight what problem you're actually trying to solve by using this mechanism, and that will highlight a more suitable approach.
-
1Thanks for your answer. Can you give a concrete examples of when "This is a Java string".hashCode() != 586653468?knorv– knorv2009年04月24日 09:20:15 +00:00Commented Apr 24, 2009 at 9:20
-
1No. Sorry. My point is that everything you test on may work the way you want. But that's still no guarantee. So if youre' working on a (say) short term project where you have control of the VM etc., then the above may work for you. But you can't rely on it in the wider world.Brian Agnew– Brian Agnew2009年04月24日 09:23:09 +00:00Commented Apr 24, 2009 at 9:23
-
2"an upgrade to a future Java version could cause problems". An upgrade to a future Java version could remove the hashCode method entirely. Or make it always return 0 for strings. That's incompatible changes for ya. The question is whether Sun^HOracle^HThe JCP would consider it a breaking change and therefore worth avoiding. Since the algorithm is in the contract, one hopes they would.Steve Jessop– Steve Jessop2009年04月24日 10:32:05 +00:00Commented Apr 24, 2009 at 10:32
-
@SteveJessop well, since
switch
statements over strings compile to code relying on a particular fixed hash code, changes toString
’s hash code algorithm would definitely break existing code...Holger– Holger2018年08月03日 17:41:39 +00:00Commented Aug 3, 2018 at 17:41
Just to answer your question and not to continue any discussions. The Apache Harmony JDK implementation seems to use a different algorithm, at least it looks totally different:
Sun JDK
public int hashCode() {
int h = hash;
if (h == 0) {
int off = offset;
char val[] = value;
int len = count;
for (int i = 0; i < len; i++) {
h = 31*h + val[off++];
}
hash = h;
}
return h;
}
Apache Harmony
public int hashCode() {
if (hashCode == 0) {
int hash = 0, multiplier = 1;
for (int i = offset + count - 1; i >= offset; i--) {
hash += value[i] * multiplier;
int shifted = multiplier << 5;
multiplier = shifted - multiplier;
}
hashCode = hash;
}
return hashCode;
}
Feel free to check it yourself...
-
23I think they're just being cool and optimizing it. :) "(multiplier << 5) - multiplier" is just 31 * multiplier, after all ...unwind– unwind2009年04月24日 11:32:13 +00:00Commented Apr 24, 2009 at 11:32
-
1But to make it clear from my side... Never rely on the hashcode because the hashcode is something internal.ReneS– ReneS2009年04月24日 12:16:04 +00:00Commented Apr 24, 2009 at 12:16
-
1what are the variables of "offset", "count" and "hashCode" mean? i suppose "hashcode" is used as a cached value, to avoid future calculations, and that "count" is the number of characters, but what is the "offset" ? suppose I wish to use this code so that it will be consistent, given a string, what should i do to it?android developer– android developer2013年11月18日 23:07:57 +00:00Commented Nov 18, 2013 at 23:07
-
@androiddeveloper See here: stackoverflow.com/questions/38546623/…Kartik Chugh– Kartik Chugh2016年12月13日 13:52:42 +00:00Commented Dec 13, 2016 at 13:52
-
1@androiddeveloper Now THAT'S an interesting question -- although I should have guessed it, based on your username. From the Android docs it looks like the contract is the same:
s[0]*31^(n-1) + s[1]*31^(n-2) + ... + s[n-1]
Unless I am mistaken, this is because Android uses Sun's implementation of the String object with no changes.Kartik Chugh– Kartik Chugh2016年12月16日 14:04:14 +00:00Commented Dec 16, 2016 at 14:04
If you're worried about changes and possibly incompatibly VMs, just copy the existing hashcode implementation into your own utility class, and use that to generate your hashcodes .
-
I was going to say this. While the other answers do answer the question, writing a separate hashCode function is probably the appropriate solution to knorv's problem.Nick H– Nick H2011年04月06日 09:38:28 +00:00Commented Apr 6, 2011 at 9:38
The hashcode will be calculated based on the ASCII values of the characters in the String.
This is the implementation in the String Class is as follows
public int hashCode() {
int h = hash;
if (h == 0 && value.length > 0) {
hash = h = isLatin1() ? StringLatin1.hashCode(value)
: StringUTF16.hashCode(value);
}
return h;
}
Collisions in hashcode are unavoidable. For example, the strings "Ea" and "FB" give the same hashcode as 2236