jdk分析之String

public class StringDemo01 {
    public static void main(String[] args) {
        String s1 = new String("abc"); // a
        String s2 = "abc"; // b
        String s3 = new String("abc"); // c

        System.out.println(s1 == s2); //false
        System.out.println(s1 == s3); //false
        System.out.println(s2 == s3); //false

        System.out.println(s1 == s1.intern()); //false
        System.out.println(s2 == s2.intern()); //true
        System.out.println(s1.intern() == s2.intern()); //true

        String lo = "lo";
        String hel = "hel";
        System.out.println("hello" == "hel" + "lo"); // d //true
        System.out.println("hello" == "hel" + lo); // e //false
        System.out.println("hello" == hel + "lo"); // f //false
    }
}

1、生成几个对象

问题:当执行完程序的第a、b、c行之后分别生成几个对象,分别是什么?原因?

1)执行完a后,生成2个对象:

首先先到String Pool中寻找是否有名称为"abc"的字符串,没有的话在String Pool中新生成一个"abc"对象1,而new String("abc")操作又会在Java内存的堆中生成一个"abc"String类对象2,

变量s1是一个引用,存放在栈中并指向堆中的"abc"String对象;

2) 执行完b后,生成0个对象:

String s2= "abc"; 此句程序在执行过程中也会先到String Pool中寻找名称为"abc"的字符串对象,因为在String s1 = new String("abc");时已经在String Pool中生成了"abc"对象,故执行完

这句之后同样存在于栈的s2对象指向String Pool中的"abc"String对象,即s2中存放"abc"String对象在String Pool中的地址;

3) 执行完c后,生成1个对象:

同样的道理,首先会到String Pool中查找是否有"abc"String对象,发现已经存在,然后又通过new的方式在中生成一个新的"abc"String对象,并将其地址赋给引用变量s3;

在此示例中三个引用变量s1、s2、s3分别指向三个不同的对象,存放着三个不同的内存地址,故三者当中任意者用"=="来比较都是false。

s1--------堆中的一个对象,此对象是String Pool中对象的一个拷贝;

s2--------String Pool常量池中的"abc"String对象;

s3--------堆中的一个对象,此对象也是String Pool中对象的一个拷贝;

==判断小结

1)对于原生数据类型来说"=="比较的是数据的数值
2)对于引用数据类型来说"=="比较的永远是对象的内存地址

2、intern

查看JDK文档可知如果有两个String类型s和t,当且仅当s.equals(t)时,s.intern()==t.intern(),即如果s和t的内容相等时s.intern()和t.intern()都是返回String Pool中同一个对象的引用,

既然是同一个对象的引用地址肯定一样。 分析s.intern()可知,首先到String Pool常量池中查找是否有"abc"字符串,如果有那么s.intern()的返回值就是常量池中"abc"的地址,如果没有,那么会先添加到string pool中去,然后将地址返回

所以可以得到如下关系:s.intern()=s1.intern()=s2.intern()=s1

s1指向的是堆,而s1.intern()指向的是常量池中;

s2指向的是常量池,s2.intern()指向的也是常量池;

s1. intern()指向的是常量池,s2.intern()指向的也是常量池;

3、字符串拼接

对于第d行"+"两端是两个字面值即常量,故会拼接成一个"hello",然后在String Pool常量池中可以找到,不生成新的对象,等号两端的地址都是String Pool中" hello"的地址,所以相等,返回true。

对于第e,f行"+"两端其中一个是字面值即常量而另一个则是变量,无法进行拼接,故会在堆中生成一个"abc"的对象,堆中一个对象和String Pool常量池中一个对象当然是两个对象,地址值肯定不相

等,所以返回false。

常量池:JVM中一块独立的区域存放字符串常量和基本类型常量(public static final)

4、在Java中一个字符串能否表示一个汉字?

在Java中一个字符表示16位,相当于两个字节(1 byte = 8 bit),而一个汉字占用两个字节。

5、String s = "Hello"; s = s + " world!";这两行代码执行后,原始的String对象中的内容到底变了没有?

String对象的内容是不能改变的,最后虽然指向了helloworld,但是原始的String对象的内容还是hello并没有改变,只是现在不指向它罢了。

6、String s = new String("xyz");创建了几个String Object? 二者之间有什么区别?

创建了2个对象,一个在string pool中一个在堆中

7、String 和StringBuffer的区别

String是final类型,内容不可改变的,StringBuffer的内容是可以改变的;

String重写了equals和hashCode方法,而StringBuffer则没有。

8、StringBuffer和StringBuilder的区别

StringBuffer是线程安全的;

StringBuilder是非线程安全的性能高点,推荐使StringBuilder,JDK5以后出现。

9、下面这条语句一共创建了多少个对象:String s="a"+"b"+"c"+"d"

一个。JVM在编译时会优化,相当于直接定义了一个”abcd”的字符串;

String s = "a"+"b"+"c";会生成几个对象?

只会生成一个对象,编译期就能确定存储到常量池中。

使用包含变量的字符串连接符如"aa" + s1创建的对象是运行期才创建的,存储在中;

10、String练习

String str1 = "a";
String str2 = "b";
String str3 = "ab";
String str4 = str1 + str2;
String str5 = new String("ab");

System.out.println(str5.equals(str3));  //true
System.out.println(str5 == str3); //false
System.out.println(str5.intern() == str3); //true
System.out.println(str5.intern() == str4); //false

String a = new String("ab");
String b = new String("ab");
String c = "ab";
String d = "a" + "b";
String e = "b";
String f = "a" + e;

System.out.println(b.intern() == a); //false
System.out.println(b.intern() == c); //true
System.out.println(b.intern() == d); //true
System.out.println(b.intern() == f); //false
System.out.println(b.intern() == a.intern()); //true

String a = "abc";
String b = "abc";
String c = "a" + "b" + "c";
String d = "a" + "bc";
String e = "ab" + "c";

System.out.println(a == b); //true
System.out.println(a == c); //true
System.out.println(a == d); //true
System.out.println(a == e); //true
System.out.println(c == d); //true
System.out.println(c == e); //true
时间: 2024-08-26 03:18:06

jdk分析之String的相关文章

JDK源码分析之String篇

------------------------------String在内存中的存储情况(一下内容摘自参考资料1)----------------------------------- 前提:先了解下什么是声明,什么时候才算是产生了对象实例 其中x并未看到内存分配,变量在使用前必须先声明,再赋值,然后才可以使用.java基础数据类型会用对应的默认值进行初始化 一.首先看看Java虚拟机JVM的内存块及其变量.对象内存空间是怎么存储分配的 1.栈:存放基本数据类型及对象变量的引用,对象本身不存放

Java之内存分析和String对象

http://www.cnblogs.com/devinzhang/archive/2012/01/25/2329463.html Java中内存分析: 栈(Stack) :存放基本类型的变量数据和对象的引用,但对象本身不存放在栈中,而是存放在堆(new 出来的对象)或者常量池中(字符串常量对象存放在常量池中). 堆(heap):存放所有new出来的对象. 常量池(constant pool):在堆中分配出来的一块存储区域,存放储显式的String常量和基本类型常量(float.int等).另外

为什么jdk中把String类设计成final

为什么jdk中把String类设计成final? 最佳答案: 主要是为了"效率" 和 "安全性" 的缘故.若 String允许被继承, 由于它的高度被使用率, 可能会降低程序的性能,所以String被定义成final. 其它答案一: String和其他基本类型不同,他是个对象类型.既然是对象类型,如果是在静态方法下是必须调用静态方法或值的,如果是非静态的方法,就必须要实例化. main函数是个static的.所以String要能像其他的基本类型一样直接被调用.这也是

【JDK源码分析】String的存储区与不可变性(转)

// ... literals are interned by the compiler // and thus refer to the same object String s1 = "abcd"; String s2 = "abcd"; s1 == s2; // --> true // ... These two have the same value // but they are not the same object String s1 = new

JDK分析工具&JVM垃圾回收(转)

转自:http://blog.163.com/[email protected]/blog/static/10510751320144201519454/ 官方手册:http://docs.oracle.com/javase/7/docs/     ---->http://docs.oracle.com/javase/7/docs/technotes/tools/solaris/java.html   java命令的各种选项的说明 参考书籍: <深入理解Java虚拟机:JVM高级特性与最佳实践

JDK源码-String

1,字符串,String类. -1,字符串是常量,他们的值在创建后不能更改.字符串缓冲区支持可变的字符串. -2,String重载了Java中的+操作. -3,String对象是不可变的,你可以给一个String对象加任意多的别名.因为String对象具有只读特性,所以指向他的任何引用都不能改变它的值. 2,String类的成员变量 -1,value 存储字符串的char数组. private final char value[]; -2,offset:存储使用的首元素index. /** Th

jdk分析工具:jps和jstack

jps 用来查看:基于HotSpot JVM里面所有进程的具体状态, 包括进程ID,进程启动的路径等等. 与unix上的ps类似,用来显示本地有权限的java进程,可以查看本地运行着几个java程序,并显示他们的进程号. 使用jps时,不需要传递进程号做为参数. Jps也可以显示远程系统上的JAVA进程,这需要远程服务上开启了jstat服务,以及RMI注及服务,不过常用都是对本地的JAVA进程的查看. 命令格式 jps [ options ] [ hostid ] 常用参数说明 -m 输出传递给

Java中String连接性能的分析

总结:如果String的数量小于4(不含4),使用String.concat()来连接String,否则首先计算最终结果的长度,再用该长度来创建一个StringBuilder,最后使用这个StringBuilder来连接所有String.      我建议大家如果确定需要连接的String的数量小于4的,直接使用String.concat()来连接,虽然StringBundler能够帮你自动处理这一情况,但创建一个String[]和那些方法调用都是一些无谓的开销. Java中的String是一个

JDK源码阅读(一):Object源码分析

最近经过某大佬的建议准备阅读一下JDK的源码来提升一下自己 所以开始写JDK源码分析的文章 阅读JDK版本为1.8 目录 Object结构图 构造器 equals 方法 getClass 方法 hashCode 方法 toString 方法 finalize 方法 registerNatives 方法 1. Object结构图 2. 类构造器 ??类构造器是创建Java对象的方法之一.一般我们都使用new关键字来进行实例,还可以在构造器中进行相应的初始化操作. ??在一个Java类中必须存在一个