java中关于String 类型数据 的存储方式
- Constant Pool常量池的概念:
- 在讲到String的一些特殊情况时,总会提到String Pool或者Constant Pool,但是我想很多人都不太
- 明白Constant Pool到底是个怎么样的东西,运行的时候存储在哪里,所以在这里先说一下Constant Pool的内容.
- String Pool是对应于在Constant Pool中存储String常量的区域.习惯称为String Pool,也有人称为
- String Constant Pool.好像没有正式的命名??
- 在java编译好的class文件中,有个区域称为Constant Pool,他是一个由数组组成的表,类型
- 为cp_info constant_pool[],用来存储程序中使用的各种常量,包括Class/String/Integer等各
- 种基本Java数据类型,详情参见The Java Virtual Machine Specification 4.4章节.
- 对于Constant Pool,表的基本通用结构为:
- cp_info {
- u1 tag;
- u1 info[];
- }
- tag是一个数字,用来表示存储的常量的类型,例如8表示String类型,5表示Long类型,info[]根据
- 类型码tag的不同会发生相应变化.
- 对于String类型,表的结构为:
- CONSTANT_String_info {
- u1 tag;
- u2 string_index;
- }
- tag固定为8,string_index是字符串内容信息,类型为:
- CONSTANT_Utf8_info {
- u1 tag;
- u2 length;
- u1 bytes[length];
- }
- tag固定为1,length为字符串的长度,bytes[length]为字符串的内容.
- (以下代码在jdk6中编译)
- 为了详细理解Constant Pool的结构,我们参看一些代码:
- String s1 = "sss111";
- String s2 = "sss222";
- System.out.println(s1 + " " + s2);
- 由于"sss111"和"sss222"都是字符串常量,在编译期就已经创建好了存储在class文件中.
- 在编译后的class文件中会存在这2个常量的对应表示:
- 08 00 11 01 00 06 73 73 73 31 31 31 08 00 13 01 ; ......sss111....
- 00 06 73 73 73 32 32 32 ; ..sss222
- 根据上面说的String常量结构,我们分析一下
- 开始的08为CONSTANT_String_info结构中的tag,而11应该是它的相对引用,01为
- CONSTANT_Utf8_info的tag,06为对应字符串的长度,73 73 73 31 31 31为字符串对
- 应的编码,接着分析,会发现后面的是对应"sss222"的存储结构.
- 经过上面分析,我们知道了11和13是两个字符串的相对引用,就可以修改class文件
- 来修改打印的内容,把class文件中的
- 00 6E 00 04 00 03 00 00 00 24 12 10 4C 12 12 4D
- 改成
- 00 6E 00 04 00 03 00 00 00 24 12 10 4C 12 10 4D
- 程序就会输出sss111 sss111,而不是和原程序一样输出sss111 sss222,因为我
- 们把对"sss222"的相对引用12改成了对"sss111"的相对引用10.
- ------------分割线
- public class Test {
- public static void main(String[] args) {
- String s1 = "sss111";
- String s2 = "sss111";
- }
- }
- 在上面程序中存在2个相同的常量"sss111",对于n个值相同的String常量,在Constant Pool中
- 只会创建一个,所以在编译好的class文件中,我们只能找到一个对"sss111"的表示:
- 000000abh: 08 00 11 01 00 06 73 73 73 31 31 31 ; ......sss111
- 在程序执行的时候,Constant Pool会储存在Method Area,而不是heap中.
- 另外,对于""内容为空的字符串常量,会创建一个长度为0,内容为空的字符串放到Constant Pool中,
- 而且Constant Pool在运行期是可以动态扩展的.
- 关于String类的说明
- 1.String使用private final char value[]来实现字符串的存储,也就是说String对象创建之后,就不能
- 再修改此对象中存储的字符串内容,就是因为如此,才说String类型是不可变的(immutable).
- 2.String类有一个特殊的创建方法,就是使用""双引号来创建.例如new String("i am")实际创建了2个
- String对象,一个是"i am"通过""双引号创建的,另一个是通过new创建的.只不过他们创建的时期不同,
- 一个是编译期,一个是运行期!
- 3.java对String类型重载了+操作符,可以直接使用+对两个字符串进行连接.
- 4.运行期调用String类的intern()方法可以向String Pool中动态添加对象.
- String的创建方法一般有如下几种
- 1.直接使用""引号创建.
- 2.使用new String()创建.
- 3.使用new String("someString")创建以及其他的一些重载构造函数创建.
- 4.使用重载的字符串连接操作符+创建.
- 例1
- /*
- * "sss111"是编译期常量,编译时已经能确定它的值,在编译
- * 好的class文件中它已经在String Pool中了,此语句会在
- * String Pool中查找等于"sss111"的字符串(用equals(Object)方法确定),
- * 如果存在就把引用返回,付值给s1.不存在就会创建一个"sss111"放在
- * String Pool中,然后把引用返回,付值给s1.
- *
- */
- String s1 = "sss111";
- //此语句同上
- String s2 = "sss111";
- /*
- * 由于String Pool只会维护一个值相同的String对象
- * 上面2句得到的引用是String Pool中同一个对象,所以
- * 他们引用相等
- */
- System.out.println(s1 == s2); //结果为true
- 例2
- /*
- * 在java中,使用new关键字会创建一个新对象,在本例中,不管在
- * String Pool中是否已经有值相同的对象,都会创建了一个新的
- * String对象存储在heap中,然后把引用返回赋给s1.
- * 本例中使用了String的public String(String original)构造函数.
- */
- String s1 = new String("sss111");
- /*
- * 此句会按照例1中所述在String Pool中查找
- */
- String s2 = "sss111";
- /*
- * 由于s1是new出的新对象,存储在heap中,s2指向的对象
- * 存储在String Pool中,他们肯定不是同一个对象,只是
- * 存储的字符串值相同,所以返回false.
- */
- System.out.println(s1 == s2); //结果为false
- 例3
- String s1 = new String("sss111");
- /*
- * 当调用intern方法时,如果String Pool中已经包含一个等于此String对象
- * 的字符串(用 equals(Object)方法确定),则返回池中的字符串.否则,将此
- * String对象添加到池中,并返回此String对象在String Pool中的引用.
- */
- s1 = s1.intern();
- String s2 = "sss111";
- /*
- * 由于执行了s1 = s1.intern(),会使s1指向String Pool中值为"sss111"
- * 的字符串对象,s2也指向了同样的对象,所以结果为true
- */
- System.out.println(s1 == s2);
- 例4
- String s1 = new String("111");
- String s2 = "sss111";
- /*
- * 由于进行连接的2个字符串都是常量,编译期就能确定连接后的值了,
- * 编译器会进行优化直接把他们表示成"sss111"存储到String Pool中,
- * 由于上边的s2="sss111"已经在String Pool中加入了"sss111",
- * 此句会把s3指向和s2相同的对象,所以他们引用相同.此时仍然会创建出
- * "sss"和"111"两个常量,存储到String Pool中.
- */
- String s3 = "sss" + "111";
- /*
- * 由于s1是个变量,在编译期不能确定它的值是多少,所以
- * 会在执行的时候创建一个新的String对象存储到heap中,
- * 然后赋值给s4.
- */
- String s4 = "sss" + s1;
- System.out.println(s2 == s3); //true
- System.out.println(s2 == s4); //false
- System.out.println(s2 == s4.intern()); //true
- 例5
- 这个是The Java Language Specification中3.10.5节的例子,有了上面的说明,这个应该不难理解了
- package testPackage;
- class Test {
- public static void main(String[] args) {
- String hello = "Hello", lo = "lo";
- System.out.print((hello == "Hello") + " ");
- System.out.print((Other.hello == hello) + " ");
- System.out.print((other.Other.hello == hello) + " ");
- System.out.print((hello == ("Hel"+"lo")) + " ");
- System.out.print((hello == ("Hel"+lo)) + " ");
- System.out.println(hello == ("Hel"+lo).intern());
- }
- }
- class Other { static String hello = "Hello"; }
- package other;
- public class Other { static String hello = "Hello"; }
- 输出结果为true true true true false true,请自行分析!
- 结果上面分析,总结如下:
- 1.单独使用""引号创建的字符串都是常量,编译期就已经确定存储到String Pool中.
- 2.使用new String("")创建的对象会存储到heap中,是运行期新创建的.
- 3.使用只包含常量的字符串连接符如"aa" + "aa"创建的也是常量,编译期就能确定,已经确定存储到String Pool中.
- 4.使用包含变量的字符串连接符如"aa" + s1创建的对象是运行期才创建的,存储在heap中.
- 6.使用"aa" + s1以及new String("aa" + s1)形式创建的对象是否加入到String Pool中我不太确定,可能是必须
- 调用intern()方法才会加入,希望高手能回答 @_@
- 还有几个经常考的面试题:
- 1.
- String s1 = new String("s1") ;
- String s2 = new String("s1") ;
- 上面创建了几个String对象?
- 答案:3个 ,编译期Constant Pool中创建1个,运行期heap中创建2个.
- 2.
- String s1 = "s1";
- String s2 = s1;
- s2 = "s2";
- s1指向的对象中的字符串是什么?
- 答案: "s1"
- //=======================================================================
- 综上:
- String str1 = "a";//constant pool
- String str2 = "b";//constant pool
- String str3 = new String("a");//heap
- System.out.println(str1 == str3);//false
- String str4 = "a"+str2;//heap
- String str5 = "a"+"b";//constant pool
- String str6 = new String("ab");//heap
- String str7 = "ab";//constant pool
- System.out.println(str4 == str5);//false
- System.out.println(str4.intern() == str5);//true
- System.out.println(str4 == str6);//false
- System.out.println(str5 == str7);//true
- String str8 = str6;//heap
- System.out.println(str6 == str8);//true
- String str9 = new String("ab");//heap
- System.out.println(str6 == str9);//false
java中关于String 类型数据 的存储方式的更多相关文章
- java 中的String类型数据添加双引号
转义符 \ 加上引号 \" <?xml version="1.0"encoding="GBK"?> String temp = &qu ...
- Java中关于String类型的一些思考
作为初学者在学习Java的时候,变量类型是不可避免会遇到的,在以往我们的印象中字符串String都是作为基本类型而存在的,但是在Java中String类型确是一个实实在在的引用类型,是可以通过new关 ...
- Java中的String类型
1.基本类型和引用类型 在C语言里面,是有指针这么一个变量类型的,指针变量保存的就是所要指向内容的地址.在Java里面,没有了指针的这么个说法,而是换了一个词:引用类型变量. 先说Java里面的基本类 ...
- Java中关于String类型的10个问题
1. 如何比较两个字符串?用“=”还是equals 简单来说,“==”是用来检测俩引用是不是指向内存中的同一个对象,而equals()方法则检测的是两个对象的值是否相等.只要你想检测俩字符串是不是相等 ...
- Java 中转换为String类型的四种方法
1. 使用 String 的构造方法,用于 byte[], char[], StringBuffer, StringBuilder 类型 2. 使用 String 的静态方法 valueOf() 推荐 ...
- Java中的String到底占用多大的内存空间?你所了解的可能都是错误的!!
写在前面 最近小伙伴加群时,我总是问一个问题:Java中的String类占用多大的内存空间?很多小伙伴的回答着实让我哭笑不得,有说不占空间的,有说1个字节的,有说2个字节的,有说3个字节的,有说不知道 ...
- 理解Java中的字符串类型
1.Java内置对字符串的支持: 所谓的内置支持,即不用像C语言通过char指针实现字符串类型,并且Java的字符串编码是符合Unicode编码标准,这也意味着不用像C++那样通过使用string和w ...
- Android java传递string类型数据给C
本文接着实现<Android java传递int类型数据给C>的还未实现的方法: public native String sayHelloInC(String s); 先贴一个工具方法, ...
- java中,字符串类型的时间数据怎样转换成date类型。
将字符串类型的时间转换成date类型可以使用SimpleDateFormat来转换,具体方法如下:1.定义一个字符串类型的时间:2.创建一个SimpleDateFormat对象并设置格式:3.最后使用 ...
随机推荐
- Git(Repo)常用命令收集
(注意: 只记录工作中实际使用的命令) 同步android源码 repo sync:(可加-c,只取当前分支: 可加-j4,线程数量) 查看android源码下所有项目的git状态 rep ...
- Python3.5入门学习记录-模块
模块让你能够有逻辑地组织你的Python代码段. 把相关的代码分配到一个 模块里能让你的代码更好用,更易懂. 模块也是Python对象,具有随机的名字属性用来绑定或引用. 简单地说,模块就是一个保存了 ...
- C++语法报错收集
1. error C2864: "OuterClass::m_outerInt": 只有静态常量整型数据成员才可以在类中初始化 class OuterClass { public: ...
- c++中的名字查找
参看下面链接:<C++中的名字查找>
- 操作数据表中的记录——SELECT (where表达式、GROUP BY、HAVING、LIMIT)
原文链接:http://www.ifyao.com/2015/01/26/%E6%93%8D%E4%BD%9C%E6%95%B0%E6%8D%AE%E8%A1%A8%E4%B8%AD%E7%9A%84 ...
- yii2 去掉index.php的方法
1.开启apache-rewrite 在Windows下,我们一般使用的是Administrator账号,所以启用这两项非常简单: 在[Apache安装目录]/conf/httpd.conf中找到 # ...
- python成长之路第三篇(4)_作用域,递归,模块,内置模块(os,ConfigParser,hashlib),with文件操作
打个广告欢迎加入linux,python资源分享群群号:478616847 目录: 1.作用域 2.递归 3.模块介绍 4.内置模块-OS 5.内置模块-ConfigParser 6.内置模块-has ...
- [转]MySQL 5.6 全局事务 ID(GTID)实现原理(二)
原文连接:http://qing.blog.sina.com.cn/1757661907/68c3cad333002qsk.html 原文作者:淘长源 转载注明以上信息 前文 MySQL 5.6 全局 ...
- Sunscreen(POJ 3614 优先队列)
Sunscreen Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 5898 Accepted: 2068 Descrip ...
- C++----练习
1.完成一个C++实现的加法程序: #include<iostream> int main() { std::cout<<"可以输入两个加数,本程序完成加法...&q ...