String a = new String ("A"); String a = "A"; 两者之间的区别?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了String a = new String ("A"); String a = "A"; 两者之间的区别?相关的知识,希望对你有一定的参考价值。

String
str
=
"a
";在栈中如果没有存放值为
"a
"的地址,等同于:
String
temp=new
String(
"a
");
String
str=temp;
关于String
str
=
"a
"的内部工作。Java内部将此语句转化为以下几个步骤:
(1)先定义一个名为str的对String类的对象引用变量:String
str;
(2)在栈中查找有没有存放值为
"a
"的地址,如果没有,则开辟一个存放字面值为
"a
"的地址,接着创建一个新的String类的对象o,并将o的字符串值指向这个地址,而且在栈中这个地址旁边记下这个引用的对象o。如果已经有了值为
"a
"的地址,则查找对象o,并返回o的地址。
(3)将str指向对象o的地址。
使用String
str
=
"a
";的方式,可以在一定程度上提高程序的运行速度,因为JVM会自动根据栈中数据的实际情况来决定是否有必要创建新对象。而对于String
str
=
new
String(
"a
");的代码,则一概在堆中创建新对象,而不管其字符串值是否相等,是否有必要创建新对象,从而加重了程序的负担
参考技术A string
s2=new
string("aa");
string
s22=s2;
//s22也指向"aa",
s22="bb";
//创建一个"bb",s22引用指向"bb",并不是改变"aa"的值,所以s2是不会受影响的
system.out.println(s2);

Java中String直接赋字符串和new String的区别 如String str=new String("a")和String str = "a"有什么区

百度的面试官问

String A="ABC";

String B=new String("ABC");

这两个值,A,B 是否相等,如果都往HashSet里面放,能放下吗?

答:(a)A==B 的判断为false;

         (b)A.equals(B)为true ;因为值相等,所以都往HashSet里面放不下,只能放一个 

 问题:==与equals()的区别:

  1. ==:比较引用类型比较的是地址值是否相同
  2. equals:比较引用类型默认也是比较地址值是否相同,注意:String类重写了equals()方法,比较的是内容是否相同。

String A = "ABC";内存会去查找永久代(常量池) ,如果没有的话,在永久代中中开辟一块儿内存空间,把地址付给栈指针,如果已经有了"ABC"的内存,直接把地址赋给栈指针;

因此

String str1="aa";

Srting str2="aa";

String Str3="aa";

....

这样下去,str1==Str2==str3;会一直相等下去,(a) ==的判断, (b) equals()的判断;都相等,因为他们的地址都相等,因此只在常量池中有一份内存空间,地址全部相同; 

  而String str = new String("a");是根据"a"这个String对象再次构造一个String对象;在堆中从新new一块儿内存,把指针赋给栈,

将新构造出来的String对象的引用赋给str。 因此 只要是new String(),则,栈中的地址都是指向最新的new出来的堆中的地址,

(a)“”==“” 是判断地址的,当然不相同;

(b)至于equals,String类型重写了 equals()方法,判断值是否相等,明显相等,因此 equals 是相等的;

这是String 重写的equals:

 * @see  #compareTo(String)
     * @see  #equalsIgnoreCase(String)
     */
    public boolean equals(Object anObject) {
        if (this == anObject) {
            return true;
        }
        if (anObject instanceof String) {
            String anotherString = (String) anObject;
            int n = value.length;
            if (n == anotherString.value.length) {
                char v1[] = value;
                char v2[] = anotherString.value;
                int i = 0;
                while (n-- != 0) {
                    if (v1[i] != v2[i])
                            return false;
                    i++;
                }
                return true;
            }
        }
        return false;
    }

 

public class StringDemo2 {
    public static void main(String[] args) {
        String s1 = new String("hello");
        String s2 = "hello";
        System.out.println(s1 == s2);// false
        System.out.println(s1.equals(s2));// true
    }
}
**运行结果:**

> false 
> true

代码详解

  1. 首先,通过main()方法进栈。
  2. 然后再栈中定义一个对象s1,去堆中开辟一个内存空间,将内存空间的引用赋值给s1,“hello”是常量,然后去字符串常量池 查看是否有hello字符串对象,没有的话分配一个空间存放hello,并且将其空间地址存入堆中new出来的空间中。
  3. 在栈中定义一个对象s2,然后去字符串常量池中查看是否有”hello”字符串对象,有,直接把”hello”的地址赋值给s2.
  4. 即s1中存的是堆中分配的空间,堆中分配的空间中存的是字符串常量池中分配空间存放”hello”的空间的地址值。而s2中之间存的是字符串常量池中分配空间存放”hello”的空间的地址值。
  5. 由于s1与s2中存放的地址不同,所以输出false。因为,类String重写了equals()方法,它比较的是引用类型的 的值是否相等,所以输出true。即结果为false、true。

Demo1

public class StringDemo1 {
    public static void main(String[] args) {
        String s1 = new String("hello");
        String s2 = new String("hello");
        System.out.println(s1 == s2);// false
        System.out.println(s1.equals(s2));// true

        String s3 = new String("hello");
        String s4 = "hello";
        System.out.println(s3 == s4);// false
        System.out.println(s3.equals(s4));// true

        String s5 = "hello";
        String s6 = "hello";
        System.out.println(s5 == s6);// true
        System.out.println(s5.equals(s6));// true
    }
}

 

Demo1详解

s1~s6用equals()的比较不解释,都是比较的值,均为true。以下讲解==

  1. s1、s2:二者均为new出来的,各自在堆中分配有空间,并各自将内存地址赋值给s1、s2。空间地址不同,==比较为false。但是各自在堆中空间中保存的值均为在字符串常量池中的同一个对象的地址。根据Demo处的图即解释不难理解。
  2. s3、s4同上Demo出解释。
  3. s5、s6都是在常量池中取值,二者都指向常量池中同一对象,其地址值相同,所以结果为true。

Demo2

public class StringDemo4 {
    public static void main(String[] args) {
        String s1 = "hello";
        String s2 = "world";
        String s3 = "helloworld";
        System.out.println(s3 == s1 + s2);// false
        System.out.println(s3.equals((s1 + s2)));// true
        System.out.println(s3 == "hello" + "world");//false
        System.out.println(s3.equals("hello" + "world"));// true
    }
}

 

Demo2详解

equals()比较方法不解释,比较值,均相等,均为true。

  1. s1与s2相加是先在字符串常量池中开一个空间,然后拼接,这个空间的地址就是s1与s2拼接后的地址。与s3的地址不同,所以输出为false。
  2. s3与”hello”+”world”作比较,”hello”+”world”先拼接成”helloworld”,然后再去字符串常量池中找是否有”helloworld”,有,所以和s3共用一个字符串对象,则为true。

总结

    1. String s = new String(“hello”)会创建2(1)个对象,String s = “hello”创建1(0)个对象。 
      注:当字符串常量池中有对象hello时括号内成立!
    2. 字符串如果是变量相加,先开空间,在拼接。
    3. 字符串如果是常量相加,是先加,然后在常量池找,如果有就直接返回,否则,就创建。

 

参考:String str=new String("a")和String str = "a"有什么区别?

以上是关于String a = new String ("A"); String a = "A"; 两者之间的区别?的主要内容,如果未能解决你的问题,请参考以下文章

String a = new String ("A"); String a = "A"; 两者之间的区别?

new String(“a”)与String a="a";

String使用字面值常量赋值与new String()赋值的区别

20170818 - 今日技能封装 - Q

String str=new String("a")和String str = "a"有什么区别?

Set<String> setStr=new HashSet<String>()