答:String 不是基础数据类型,它是从堆上分配来的。基础数据类型有 8 个,分别为:boolean、byte、short、int、long、float、double、char。
答:str.length()
题目解析:字符串没有 length 属性,只有 length()
方法。
答:"==" 对基本类型来说是值比较,对于引用类型来说是比较的是引用;而 equals 默认情况下是引用比较,只是很多类重写了 equals 方法,比如 String、Integer 等把它变成了值比较,所以一般情况下 equals 比较的是值是否相等。
① "==" 解读
对于基本类型和引用类型 == 的作用效果是不同的,如下所示:
代码示例:
String x = "string"; String y = "string"; String z = new String("string"); System.out.println(x==y); // true System.out.println(x==z); // false System.out.println(x.equals(y)); // true System.out.println(x.equals(z)); // true
代码说明:因为 x 和 y 指向的是同一个引用,所以 ==
也是 true,而 new String()
方法则重写开辟了内存空间,所以 ==
结果为 false,而 equals 比较的一直是值,所以结果都为 true。
② equals 解读
equals 本质上就是 ==
,只不过 String 和 Integer 等重写了 equals 方法,把它变成了值比较。看下面的代码就明白了。
首先来看默认情况下 equals 比较一个有相同值的对象,代码示例:
class Cat { public Cat(String name) { this.name = name; } private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } } Cat c1 = new Cat("红烧肉"); Cat c2 = new Cat("红烧肉"); System.out.println(c1.equals(c2)); // false
输出结果出乎我们的意料,竟然是 false?!
这是怎么回事,看了 equals 源码就知道了,源码如下:
public boolean equals(Object obj) { return (this == obj); }
原来 equals 本质上就是 ==
。
那问题来了,两个相同值的 String 对象,为什么返回的是 true?代码如下:
String s1 = new String("老王"); String s2 = new String("老王"); System.out.println(s1.equals(s2)); // true
同样的,当进入 String 找到其 equals 方法,就能发现答案,代码如下:
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; }
原来是 String 重写了 Object 的 equals 方法,把引用比较改成了值比较。
总结来说,"==" 对于基本类型来说是值比较,对于引用类型来说是比较的是引用;而 equals 默认情况下是引用比较,只是很多类重写了 equals 方法,比如 String、Integer 等把它变成了值比较,所以一般情况下 equals 比较的是值是否相等。(重点注意)
String str = "laowang"; str.substring(0,1); System.out.println(str);
答:laowang
题目解析:因为 String 的 substring() 方法不会修改原字符串内容,所以结果还是 laowang。
String s1 = "hi," + "lao" + "wang"; String s2 = "hi,"; s2 += "lao"; s2 += "wang"; String s3 = "hi,laowang"; System.out.println(s1 == s2); System.out.println(s1 == s3); System.out.println(s2 == s3);
答:false true false
题目解析:String s1 = "hi," + "lao" + "wang" 代码会被 JVM 优化为:String s1 = "hi,laowang",这样就和 s3 完全相同,s1 创建的时候会把字符"hi,laowang"放入常量池,s3 创建的时候,常量池中已经存在对应的缓存,会直接把引用返回给 s3,所以 s1==s3
就为 true,而 s2 使用了 +=
其引用地址就和其他两个不同。
public static void main(String[] args) { String str = new String("laowang"); change(str); System.out.println(str); } public static void change(String str) { str = "xiaowang"; }
答:laowang
public static void main(String[] args) { StringBuffer sf = new StringBuffer("hi,"); changeSf(sf); System.out.println(sf); } public static void changeSf(StringBuffer sf){ sf.append("laowang"); }
答:hi,laowang
题目解析:String 为不可变类型,在方法内对 String 修改的时候,相当修改传递过来的是一个 String 副本,所以 String 本身的值是不会被修改的,而 StringBuffer 为可变类型,参数传递过来的是对象的引用,对其修改它本身就会发生改变。
String str = "abcdef";
System.out.println(str.substring(3, 3));
答:""(空)。(这里记忆与python列表切分相同,栅栏放前面)
答:常用的方式有以下两种。
答:以下是 String、StringBuffer、StringBuilder 的区别:
答:intern() 方法用于查找常量池中是否存在该字符值,如果常量池中不存在则先在常量池中创建,如果已经存在则直接返回。
代码示例:
String s = "laowang"; String s2 = s.intern(); System.out.println(s == s2); // 返回 true
答:创建了一个或两个对象,如果常量池中已经有了字符串 “laowang”,就只会创建一个引用对象 s 指向常量池中的对象 ”laowang“;如果常量池中没有字符串 ”laowang“,则先会在常量池中创建一个对象 ”laowang“,再创建一个引用对象 s 指向常量池中的对象,所以答案是创建一个或者两个对象。
字符串常量池是存储在 Java 堆内存中的字符串池,是为防止每次新建字符串带的时间和空间消耗的一种解决方案。在创建字符串时 JVM 会首先检查字符串常量池,如果字符串已经存在池中,就返回池中的实例引用,如果字符串不在池中,就会实例化一个字符串放到池中并把当前引用指向该字符串。
答:不可变的好处如下。
答:String 不能被继承。因为 String 被声明为 final(最终类),所以不能被继承,源码如下(JDK 8)。
public final class String implements java.io.Serializable, Comparable<String>, CharSequence { //...... }
2.2号Java复习题目——Java的字符串 (面试题刷题)
原文:https://www.cnblogs.com/LeoLRH/p/12254318.html