1、判断s1和s2是否相等
String s1 = "123"; String s2 = "123"; System.out.println(s1 == s2); // true System.out.println(s1.equals(s2)); // true
String字符串属于常量,常量需要进入内存中的字符串常量池(进入常量池规则:如果常量池中没有这个常量,就创建一个,如果有就不再创建了),所以s2的引用也指向内存区中已经存在的“123”,所以地址一样,==对于引用类型比较的是地址,String类和其他基本类型的封装类中重写了equals方法,比较的是值。
2、以下代码创建了几个对象
String s = new String("abc");
2个对象(常量池中没“abc”对象),如果常量池中有abc,则是创建1个对象。s首先会在方法区的常量池创建“abc”字符串对象,当new的时候就会在堆内存中创建一个对象,此时会把常量池中的字符串常量拷贝一份副本给堆内存中的对象,堆内存中的对象会把地址赋值给s。
常量池中的对象的地址和堆内存中对象的地址是不一样的,故创建2个对象。
3、判断new String是否相等
String s1 = new String("abc"); String s2 = "abc"; System.out.println(s1 == s2); // false System.out.println(s1.equals(s2)); // true
根据上一题得出,执行完s1后,内存中会有两个对象,一个在方法区中的常量池,一个在堆内存,地址值是不一样的,s1指向的是堆内存中的对象,当执行到s2时,s2指向的是方法区常量池中的对象。地址值不一样,==比较为false,String类重写equals方法,比较的值,故true
4、string被final修饰,值是不可变的,解释?
String s = "Java";
s = "HTML";
第一条语句创建了一个内容为"Java"的String对象,并将其引用赋值给s。
第二条语句创建了一个内容为"HTML"的新String对象,并将其引用赋值给s。
赋值后第一个String对象仍然存在,但是不能再访问它,因为变量s现在指向了新的对象。
5、拼接字符串是否相等
String s1 = "a"+"b"+"c"; String s2 = "abc"; System.out.println(s1 == s2); // true
Java中有常量优化机制,“a”、“b”、“c”本身就是字符串常量,所以在编译时,“a”+“b”+"c"就是“abc”字符串,所以就在常量池创建了“abc”字符串,当执行s2的时候,此时常量池中已经存在了“abc”,所以==号比较返回true。equals方法比较毫无疑问是true。
6、变量拼接字符串相等吗?
String s1 = "ab"; String s2 = "abc"; String s3 = s1 + "c"; System.out.println(s3 == s2); // false System.out.println(s3.equals(s2)); // true
s1+"c"中s1不是常量,涉及到变量相加,所以会生成新的对象,其内部实现是先new一个StringBuilder/StringBuffer,然后 append(s1);append(“c”);然后让s3引用toString()返回的这个对象。
s3指向堆内存中的对象,s2是方法区常量池中的对象,故为false;
7.接上边第6条,阿里巴巴java开发规范明确不让for循环中使用"+"拼接字符串,原因如下:
public class demo { public static void main(String args[]) { //string的for循环用"+"拼接字符串,耗时20681ms long start = System.currentTimeMillis(); String str = "start"; for (int i = 0; i < 100000; i++) { str = str + "hello"; /* 其实"+"号拼接字符串,等价于下面这三行代码 StringBuilder sb = new StringBuilder(); sb.append(str).append("hello"); str = sb.toString(); */ } long end = System.currentTimeMillis(); System.out.println(end - start + "ms");//20681ms //建议写法 long start1 = System.currentTimeMillis(); String str1 = "start"; StringBuilder sb = new StringBuilder(); sb.append(str1); for (int i = 0; i < 10000; i++) { sb.append("hello"); } long end1 = System.currentTimeMillis(); System.out.println(end1 - start1 + "ms");//13ms } }