Java常量池的一些思考……
2012-12-03 22:29
134 查看
(此博文已移动到.net学习栏目下,主要还是想取消java栏目,因为我目前的精力主要放在.net学习上。为何敢移动呢?因为常量池这块c#和java在思想上是相通的……)
为什么考虑到常量池的问题,是因为在对于Java中return一个字符串的内存分析时想到的。
举例:
public String getInfo() {
return "Name:" + getName() + "\nAge:" + getAge() + "\nSchool" + school;
}//仅是举这样一个例子,在方法getInfo()中返回一个字符串。
System.out.println(person.getInfo()); //这也是举一个例子,即调用getInfo().
内存分析:当调用getInfo()时,会在data segment(数据区)中存放字符串"Name:" + getName() + "\nAge:" + getAge() + "\nSchool" + school;
然后在栈内存中开辟一块内存区域,让这块内存区域指向data segment中的这个字符串。当打印完后栈内存的那块引用就会消失,然后数据区中的字符串会等待垃圾回收。
有关常量池的一些思考:(里边一些想法也是借鉴了一些大神们的观点,特此声明,向他们致敬!)。
容易混淆的概念:我觉得常量池即为data segment ,虽然好多大神都说常量池在栈中,但是经过各种的内存分析,觉得常量池和data segment都是一样的内存分析。此处有大神觉得不对的地方,可以对我指教一下!嘿嘿……
常量池:(我吧,主要是讲一些有关String跟常量池有关内存分析的一些关系!)
1:String 对象本身:对于new(运行期)出来的对象,将存放于堆中。对于字符串常量(即在编译期已创建好的,即用双引号“”定义的)对象存放在常量池中。
2:String s1 = "abc";
String s2 = new String("abc");//第二种是用new()来新建的对象,它会存放在堆中。
3:对于通过new产生一个字符串,会先去常量池中查找是否已有“abc”对象,若没有则在常量池中创建一个此String对象,然后堆中再创建一个常量池中“abc”对象的拷贝对象。
4:String s1 = "abc"; //abc的值在程序编译为.class文件后,就在.class文件中生成了。
5:执行java程序的过程:(1):.class文件生成(2)被jvm装载到内存执行。
6:String s1 = "abc" + 1; //JVM对于String常量的“+”号连接,编译器在.class文件中就已经是"abc1",在编译期其字符串常量的值就确定下来。
7:String s1 = "abc";
String s2 = "def";
String s2 = s2 + "java";
//s2先指向一个常量,内容为def,但此时s2不再指向原来的那个值,而是指向了字符串常量“defjava”,原先变量还存于内存中,等待垃圾回收。
8:String s1 = "abc";注:对象可能并没有被创建,而可能只是指向一个先前已被创建的一个对象而已。
好了,就是有一些这种感想,其实我觉得对于java常量池,按照data segment的思路去分析内存,应该就可以了……本来常量池也是一个很复杂的问题……就这样吧……
为什么考虑到常量池的问题,是因为在对于Java中return一个字符串的内存分析时想到的。
举例:
public String getInfo() {
return "Name:" + getName() + "\nAge:" + getAge() + "\nSchool" + school;
}//仅是举这样一个例子,在方法getInfo()中返回一个字符串。
System.out.println(person.getInfo()); //这也是举一个例子,即调用getInfo().
内存分析:当调用getInfo()时,会在data segment(数据区)中存放字符串"Name:" + getName() + "\nAge:" + getAge() + "\nSchool" + school;
然后在栈内存中开辟一块内存区域,让这块内存区域指向data segment中的这个字符串。当打印完后栈内存的那块引用就会消失,然后数据区中的字符串会等待垃圾回收。
有关常量池的一些思考:(里边一些想法也是借鉴了一些大神们的观点,特此声明,向他们致敬!)。
容易混淆的概念:我觉得常量池即为data segment ,虽然好多大神都说常量池在栈中,但是经过各种的内存分析,觉得常量池和data segment都是一样的内存分析。此处有大神觉得不对的地方,可以对我指教一下!嘿嘿……
常量池:(我吧,主要是讲一些有关String跟常量池有关内存分析的一些关系!)
1:String 对象本身:对于new(运行期)出来的对象,将存放于堆中。对于字符串常量(即在编译期已创建好的,即用双引号“”定义的)对象存放在常量池中。
2:String s1 = "abc";
String s2 = new String("abc");//第二种是用new()来新建的对象,它会存放在堆中。
3:对于通过new产生一个字符串,会先去常量池中查找是否已有“abc”对象,若没有则在常量池中创建一个此String对象,然后堆中再创建一个常量池中“abc”对象的拷贝对象。
4:String s1 = "abc"; //abc的值在程序编译为.class文件后,就在.class文件中生成了。
5:执行java程序的过程:(1):.class文件生成(2)被jvm装载到内存执行。
6:String s1 = "abc" + 1; //JVM对于String常量的“+”号连接,编译器在.class文件中就已经是"abc1",在编译期其字符串常量的值就确定下来。
7:String s1 = "abc";
String s2 = "def";
String s2 = s2 + "java";
//s2先指向一个常量,内容为def,但此时s2不再指向原来的那个值,而是指向了字符串常量“defjava”,原先变量还存于内存中,等待垃圾回收。
8:String s1 = "abc";注:对象可能并没有被创建,而可能只是指向一个先前已被创建的一个对象而已。
好了,就是有一些这种感想,其实我觉得对于java常量池,按照data segment的思路去分析内存,应该就可以了……本来常量池也是一个很复杂的问题……就这样吧……
相关文章推荐
- 关于刚进职场的程序员的一些思考
- [置顶] 关于产品的一些思考——腾讯之UIDesigner
- 关于MQ编码(算术编码、JPEG2000、JBIG2)常见问题的一些思考
- 关于路径的一些思考
- 0302思考并回答一些问题
- 对IT行业的一些思考
- 关于Java里static的一些思考
- [转帖] 关于sizeof()的一些思考
- 有关表达式求值的一些思考
- WEB开发时的一些思考
- [体感游戏]关于体感游戏的一些思考(六)--- 飞行
- 在经营景城网过程中对alax排名的一些思考
- 一点感悟、一些思考
- 测试管理_关于测试部建设的一些思考(数据体现)
- 一些需要思考的小故事
- 对javascript的一些思考
- 一些关于面向对象设计的思考
- pull request的一些思考
- 一些关于学习和思维的思考
- 非ROOT实现静默安装的一些思考与体会,AIDL获取IPackageManager,反射ServiceManager,系统签名