论坛首页 Java企业应用论坛

Java堆.栈和常量池 笔记

浏览 58804 次
该帖已经被评为良好帖
作者 正文
   发表时间:2010-10-14  
zjhzjf000 写道
qiren83 写道
很久没看到这么高质量的贴子 学到很多

比起有人在CSDN和JY上天天讨论 String i = "ddd"+new String("KK");
成生几个对象 这贴子站在更高的层面上了

我收藏+收藏夹+右键另存为

好帖!



那位帅哥美女 说说到底是几个啊  我不知道啊



4个
"kk", new.., "ddd", i

先看"kk",在常量池中生成第一个,在堆中new了一个String对象指向常量池中的"kk","ddd"道理同"kk",最后看
最左边,在栈中声明了一个String 对象 i 分别指向常量池中的"ddd" 和 堆中的new出来的String对象,
可能说的不太清楚,有错请指出,谢谢
0 请登录后投票
   发表时间:2010-10-14  
yqfly2008 写道
zjhzjf000 写道
qiren83 写道
很久没看到这么高质量的贴子 学到很多

比起有人在CSDN和JY上天天讨论 String i = "ddd"+new String("KK");
成生几个对象 这贴子站在更高的层面上了

我收藏+收藏夹+右键另存为

好帖!



那位帅哥美女 说说到底是几个啊  我不知道啊



4个
"kk", new.., "ddd", i

先看"kk",在常量池中生成第一个,在堆中new了一个String对象指向常量池中的"kk","ddd"道理同"kk",最后看
最左边,在栈中声明了一个String 对象 i 分别指向常量池中的"ddd" 和 堆中的new出来的String对象,
可能说的不太清楚,有错请指出,谢谢

错了! 对于"+"操作,jvm会在堆中创建一个StringBuilder类,并使用堆中new出来的String对象指向返字符串"kk"进行初始化,通过调用其append方法对"ddd"进行合并操作,最后调用toString方法返回这个新的String对象个栈中的String对象i,所以这里还有一个StringBuilder对象,一共是5个对象
唉,,,感觉自己基础还是不扎实,老遗忘一些东西
0 请登录后投票
   发表时间:2010-10-14  
yqfly2008 写道
zjhzjf000 写道
qiren83 写道
很久没看到这么高质量的贴子 学到很多

比起有人在CSDN和JY上天天讨论 String i = "ddd"+new String("KK");
成生几个对象 这贴子站在更高的层面上了

我收藏+收藏夹+右键另存为

好帖!



那位帅哥美女 说说到底是几个啊  我不知道啊



4个
"kk", new.., "ddd", i

先看"kk",在常量池中生成第一个,在堆中new了一个String对象指向常量池中的"kk","ddd"道理同"kk",最后看
最左边,在栈中声明了一个String 对象 i 分别指向常量池中的"ddd" 和 堆中的new出来的String对象,
可能说的不太清楚,有错请指出,谢谢

4个应该是没有错的,但是楼上说的 "在栈中声明了一个String 对象 i 分别指向常量池中的"ddd" 和 堆中的new出来的String对象" 好象不太正确,JDK文档中说String一旦创建了就不能改变,我个人感觉应是 "ddd"+new String("kk")在堆里重新创建了一个String Object,i就指向这个新的Object.
(个人理解不对的地方大家说说,防止误导他人)
0 请登录后投票
   发表时间:2010-10-14  
个人觉得,堆里面的 new 字符串,好像不是在堆里面的。堆里面只存了一个引用。
这个引用还是指向常量池里面的字符串。先由对象引用指向堆,然后由堆指向常量池
我的个人认为,如有不对,请多多提醒。
0 请登录后投票
   发表时间:2010-10-14  
.寄存器:最快的存储区, 由编译器根据需求进行分配,我们在程序中无法控制.

-------------------------------------------------------------

这个最快就有点商榷了,  CPU的寄存器是最快的美错。JVM寄存器也只是内存的一片区域吧?
0 请登录后投票
   发表时间:2010-10-15  
gteam.yu 写道
yqfly2008 写道
zjhzjf000 写道
qiren83 写道
很久没看到这么高质量的贴子 学到很多

比起有人在CSDN和JY上天天讨论 String i = "ddd"+new String("KK");
成生几个对象 这贴子站在更高的层面上了

我收藏+收藏夹+右键另存为

好帖!



那位帅哥美女 说说到底是几个啊  我不知道啊



4个
"kk", new.., "ddd", i

先看"kk",在常量池中生成第一个,在堆中new了一个String对象指向常量池中的"kk","ddd"道理同"kk",最后看
最左边,在栈中声明了一个String 对象 i 分别指向常量池中的"ddd" 和 堆中的new出来的String对象,
可能说的不太清楚,有错请指出,谢谢

4个应该是没有错的,但是楼上说的 "在栈中声明了一个String 对象 i 分别指向常量池中的"ddd" 和 堆中的new出来的String对象" 好象不太正确,JDK文档中说String一旦创建了就不能改变,我个人感觉应是 "ddd"+new String("kk")在堆里重新创建了一个String Object,i就指向这个新的Object.
(个人理解不对的地方大家说说,防止误导他人)

我和你想法一样:4个:常量池中3个:"ddd","KK","dddKK".堆中一个:new String("KK")。

0 请登录后投票
   发表时间:2010-10-17  
如果说是String对象的话,4个是没错,全面的说,在整个过程中,“对象”是有5个的,楼上说的4个String对象,还有就是“+”操作时产生的StringBuilder对象
0 请登录后投票
   发表时间:2010-11-11  
很透彻,曾经在这迷茫了好长时间。。谢楼主
0 请登录后投票
   发表时间:2010-11-12  
虽然以前看过,但是没有整理过,学习!
0 请登录后投票
   发表时间:2010-12-11  
longxiaoyan 写道

今天复习了一下这些知识,顺便做了下笔记.
1.寄存器:最快的存储区, 由编译器根据需求进行分配,我们在程序中无法控制.
2. 栈:存放基本类型的变量数据和对象的引用,但对象本身不存放在栈中,而是存放在堆(new 出来的对象)或者常量池中(字符串常量对象存放在常量池中。)
3. 堆:存放所有new出来的对象。
4. 静态域:存放静态成员(static定义的)
5. 常量池:存放字符串常量和基本类型常量(public static final)。
6. 非RAM存储:硬盘等永久存储空间

这里我们主要关心栈,堆和常量池,对于栈和常量池中的对象可以共享,对于堆中的对象不可以共享。栈中的数据大小和生命周期是可以确定的,当没有引用指向数据时,这个数据就会消失。堆中的对象的由垃圾回收器负责回收,因此大小和生命周期不需要确定,具有很大的灵活性。
对于字符串:其对象的引用都是存储在栈中的,如果是编译期已经创建好(直接用双引号定义的)的就存储在常量池中,如果是运行期(new出来的)才能确定的就存储在堆中。对于equals相等的字符串,在常量池中永远只有一份,在堆中有多份。
如以下代码:

	String s1 = "china";
	String s2 = "china";
	String s3 = "china";
	String ss1 = new String("china");
	String ss2 = new String("china");
	String ss3 = new String("china");


这里解释一下黄色这3个箭头,对于通过new产生一个字符串(假设为”china”)时,会先去常量池中查找是否已经有了”china”对象,如果没有则在常量池中创建一个此字符串对象,然后堆中再创建一个常量池中此”china”对象的拷贝对象。这也就是有道面试题:String s = new String(“xyz”);产生几个对象?一个或两个,如果常量池中原来没有”xyz”,就是两个。



对于基础类型的变量和常量:变量和引用存储在栈中,常量存储在常量池中。
如以下代码:

	int i1 = 9;
	int i2 = 9;
	int i3 = 9;	
	public static final int INT1 = 9;
	public static final int INT2 = 9;
	public static final int INT3 = 9;




对于成员变量和局部变量:成员变量就是方法外部,类的内部定义的变量;局部变量就是方法或语句块内部定义的变量。局部变量必须初始化。
形式参数是局部变量,局部变量的数据存在于栈内存中。栈内存中的局部变量随着方法的消失而消失。
成员变量存储在堆中的对象里面,由垃圾回收器负责回收。
如以下代码:

class BirthDate {
    private int day;
    private int month;
    private int year;    
    public BirthDate(int d, int m, int y) {
        day = d; 
        month = m; 
        year = y;
    }
    省略get,set方法………
}

public class Test{
    public static void main(String args[]){
int date = 9;
        Test test = new Test();      
		   test.change(date); 
        BirthDate d1= new BirthDate(7,7,1970);       
    }  

    public void change1(int i){
    	i = 1234;
    }



}


对于以上这段代码,date为局部变量,i,d,m,y都是形参为局部变量,day,month,year为成员变量。下面分析一下代码执行时候的变化:
1. main方法开始执行:int date = 9;
date局部变量,基础类型,引用和值都存在栈中。
2. Test test = new Test();
test为对象引用,存在栈中,对象(new Test())存在堆中。
3. test.change(date);
i为局部变量,引用和值存在栈中。当方法change执行完成后,i就会从栈中消失。
4. BirthDate d1= new BirthDate(7,7,1970); 
d1为对象引用,存在栈中,对象(new BirthDate())存在堆中,其中d,m,y为局部变量存储在栈中,且它们的类型为基础类型,因此它们的数据也存储在栈中。day,month,year为成员变量,它们存储在堆中(new BirthDate()里面)。当BirthDate构造方法执行完之后,d,m,y将从栈中消失。
5.main方法执行完之后,date变量,test,d1引用将从栈中消失,new Test(),new BirthDate()将等待垃圾回收。

 

 

0 请登录后投票
论坛首页 Java企业应用版

跳转论坛:
Global site tag (gtag.js) - Google Analytics