当前位置:   article > 正文

java面试强基(9)_“+”和“+=”是java中仅有的两个运算符重载。

“+”和“+=”是java中仅有的两个运算符重载。

 

字符串拼接用“+” 还是 StringBuilder?

Java 语言本身并不支持运算符重载,“+”和“+=”是专门为 String 类重载过的运算符,也是 Java 中仅有的两个重载过的运算符。

​ 字符串对象通过“+”的字符串拼接方式,实际上是通过 StringBuilder 调用 append() 方法实现的,拼接完成之后调用 toString() 得到一个 String 对象 。

String#equals() 和 Object#equals() 有何区别? 

​ String 中的 equals 方法是被重写过的,比较的是 String 字符串的值是否相等。 Object 的 equals 方法是比较的对象的内存地址。

字符串常量池的作用了解吗? 

​ 字符串常量池 是 JVM 为了提升性能和减少内存消耗针对字符串(String 类)专门开辟的一块区域,主要目的是为了避免字符串的重复创建。

String s1 = new String("abc");这句话创建了几个字符串对象? 

会创建 1 或 2 个字符串对象。

​ 1、如果字符串常量池中不存在字符串对象“abc”的引用,那么会在堆中创建 2 个字符串对象“abc”。

​ 2、如果字符串常量池中已存在字符串对象“abc”的引用,则只会在堆中创建 1 个字符串对象“abc”。

 

intern 方法有什么作用? 

​ String.intern() 是一个 native(本地)方法,其作用是将指定的字符串对象的引用保存在字符串常量池中,可以简单分为两种情况:

  • 如果字符串常量池中保存了对应的字符串对象的引用,就直接返回该引用。
  • 如果字符串常量池中没有保存了对应的字符串对象的引用,那就在常量池中创建一个指向该字符串对象的引用并返回。

尽管在输出中调用intern方法并没有什么效果,但是实际上后台这个方法会做一系列的动作和操作。在调用”ab”.intern()方法的时候会返回”ab”,但是这个方法会首先检查字符串池中是否有”ab”这个字符串,如果存在则返回这个字符串的引用,否则就将这个字符串添加到字符串池中,然会返回这个字符串的引用。

  1. String str1 = "a";
  2. String str2 = "b";
  3. String str3 = "ab";
  4. String str4 = str1 + str2;
  5. String str5 = new String("ab");
  6. System.out.println(str5.equals(str3));
  7. System.out.println(str5 == str3);
  8. System.out.println(str5.intern() == str3);
  9. System.out.println(str5.intern() == str4);

 

为什么会得到这样的一个结果呢?我们一步一步的分析。

  •  第一、str5.equals(str3)这个结果为true,不用太多的解释,因为字符串的值的内容相同。
  •  第二、str5 == str3对比的是引用的地址是否相同,由于str5采用new String方式定义的,所以地址引用一定不相等。所以结果为false。
  •  第三、当str5调用intern的时候,会检查字符串池中是否含有该字符串。由于之前定义的str3已经进入字符串池中,所以会得到相同的引用。
  •  第四,当str4 = str1 + str2后,str4的值也为”ab”,但是为什么这个结果会是false呢?先看下面代码:
  1. String a = new String("ab");
  2. String b = new String("ab");
  3. String c = "ab";
  4. String d = "a" + "b";
  5. String e = "b";
  6. String f = "a" + e;
  7. System.out.println(b.intern() == a);
  8. System.out.println(b.intern() == c);
  9. System.out.println(b.intern() == d);
  10. System.out.println(b.intern() == f);
  11. System.out.println(b.intern() == a.intern());

 

由运行结果可以看出来,b.intern() == a和b.intern() == c可知,采用new 创建的字符串对象不进入字符串池,并且通过b.intern() == d和b.intern() == f可知,字符串相加的时候,都是静态字符串的结果会添加到字符串池,如果其中含有变量(如f中的e)则不会进入字符串池中。但是字符串一旦进入字符串池中,就会先查找池中有无此对象。如果有此对象,则让对象引用指向此对象如果无此对象,则先创建此对象,再让对象引用指向此对象。 

当研究到这个地方的时候,突然想起来经常遇到的一个比较经典的Java问题,就是对比equal和==的区别,当时记得老师只是说“==”判断的是“地址”,但是并没说清楚什么时候会有地址相等的情况。现在看来,在定义变量的时候赋值,如果赋值的是静态的字符串,就会执行进入字符串池的操作,如果池中含有该字符串,则返回引用。 

 

  1. String a = "abc";
  2. String b = "abc";
  3. String c = "a" + "b" + "c";
  4. String d = "a" + "bc";
  5. String e = "ab" + "c";
  6. System.out.println(a == b);
  7. System.out.println(a == c);
  8. System.out.println(a == d);
  9. System.out.println(a == e);
  10. System.out.println(c == d);
  11. System.out.println(c == e);

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/我家小花儿/article/detail/468068
推荐阅读
相关标签
  

闽ICP备14008679号