Manipulation of a string in constant pool vs Manipulation of a string in non-constant pool - Perfomance

荒凉一梦 提交于 2019-12-12 06:38:56

问题


I wanted to change a code which was like

if("T".equalsIgnoreCase(bo.getSubjectType()))  //method-1

to

if(String.valueOf('T').equalsIgnoreCase(bo.getSubjectType()))  //method-2

So for that reason I wrote a sample code

String s1 = "T";
String s2 = "T";
char c1 = 'T';
char c2 = 'T';
System.out.println(String.valueOf(c1) == String.valueOf(c2));   //  false
System.out.println(s1 == s2);   //  true

Which pretty much says String.valueOf(arg) produces a String literal and places it in constant pool. So my question is whether there would be any contrast in performance when we try to manipulate a String in non-constant pool and that in a constant pool - basically which one would be better approach method-1 or method-2?


回答1:


if(String.valueOf('T').equalsIgnoreCase(bo.getSubjectType()))

There is no advantage whatsoever to writing it like this.

String.valueOf('T') will always return a new string, equal to "T", because the result of String.valueOf(char) (or any valueOf overload, for that matter) is not cached. There is no reason to keep on creating the same string over and over.

Plus, it's more verbose.

Just stick with method 1.




回答2:


Method 2 is hard-readable and has no improvements at all.

String created using new operator operator, it always creates a new object in heap memory. String created using String literal may return an existing object from the String pool, if it already exists.

It will not return a String from the pool and creates a String using new operator (Oracle JDK 10 source):

 /**
 * Returns the string representation of the {@code char}
 * argument.
 *
 * @param   c   a {@code char}.
 * @return  a string of length {@code 1} containing
 *          as its single character the argument {@code c}.
 */
public static String valueOf(char c) {
    if (COMPACT_STRINGS && StringLatin1.canEncode(c)) {
        return new String(StringLatin1.toBytes(c), LATIN1);
    }
    return new String(StringUTF16.toBytes(c), UTF16);
}

If you want to define a String constant and that always will be loaded from the pool, just create a:

public static final String T = "T";

From JLS about string literals and pools:

class Test {
   public static void main(String[] args) {
   String hello = "Hello", lo = "lo";
   System.out.print((hello == "Hello") + " ");
   System.out.print((Other.hello == hello) + " ");
   System.out.print((other.Other.hello == hello) + " ");
   System.out.print((hello == ("Hel"+"lo")) + " ");
   System.out.print((hello == ("Hel"+lo)) + " ");
   System.out.println(hello == ("Hel"+lo).intern());
 }
}
class Other { static String hello = "Hello"; }

and the compilation unit:

package other; 
public class Other { 
    public static String hello = "Hello"; 
}

produces the output:

true true true true false true

This example illustrates six points:

• Literal strings within the same class (§8 (Classes)) in the same package (§7 (Packages)) represent references to the same String object (§4.3.1).

• Literal strings within different classes in the same package represent references to the same String object.

• Literal strings within different classes in different packages likewise represent references to the same String object.

• Strings computed by constant expressions (§15.28) are computed at compile time and then treated as if they were literals.

• Strings computed by concatenation at run time are newly created and therefore distinct.

• The result of explicitly interning a computed string is the same string as any pre-existing literal string with the same contents.



来源:https://stackoverflow.com/questions/54343097/manipulation-of-a-string-in-constant-pool-vs-manipulation-of-a-string-in-non-con

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!