1、java中所有变量(包括你上述提及的str引用类型变量)的存放位置都取决于该 变量的声明位置,而New出的对象则始终被JVM存放在堆中,创建的字串常 量则始终被JVM存放在数据段区常量池中。2、你对第一个str引用变量存放位置的解释有一点欠妥的地方,准确的说这个 str的存放位置不一定在栈中,这要看str...
java 堆栈问题
献丑了:
1、java中所有变量(包括你上述提及的str引用类型变量)的存放位置都取决于该
变量的声明位置,而New出的对象则始终被JVM存放在堆中,创建的字串常
量则始终被JVM存放在数据段区常量池中。
2、你对第一个str引用变量存放位置的解释有一点欠妥的地方,准确的说这个
str的存放位置不一定在栈中,这要看str的声明位置,如果str是方法参数中
声明或方法内局部变量声明则存放位置就是你说的在栈中,但如果在类的成
员变量中声明则是存放于堆中。“abc”是一个字串常量被存放于常量池中而
该常量返回的引用地址被存放于str变量中,str变量的位置取决于str的声明
位置。
3、第二种方式String str = new String("abc");其实上是创建了两个对象,
与上述相同,字串常量对象被放在常量池中,这个常量“abc”与上面第一个
创建的“abc”是同一个abc,常量池中不能保存相同面值的常量,而New出的
新对象new String("abc")则存放在堆中,这里的“abc”的位置存在于堆
中,与上述常量池中的“abc”不是同一个“abc”,这里一定要注意啊,而
str变量的存放位置与第一个str的存放位置都是取决于该str的声明位置。
4、解释引用变量存放位置的原理:对象在创建时将为所有的成员变量分配内存
空间以及为所有的成员方法分配入口地址,因为对象是对属性及方法的封装
而对象在创建时被分配到堆中的空间,因此成员变量被分配在堆中,而方法
时在调用时为局部参数或变量临时分配内存空间,在此,方法在调用时被JVM
线程加载至栈中,于此局部变量及参数的所在位置存在与栈中,这些局部变
量及参数所占空间是临时的,一旦方法调用结束其临时空间将被释放,所以
内存栈区空间是为变量分配的临时存贮空间,这个你可以参考马士兵的堆栈
视屏教程,他讲述的很清楚,我可能还没有他表达得清楚。
5、最后再强调一点:变量不是都放在栈中,这取决于该变量的声明位置,我的
这种说法你能理解吗?只有理解了这一点,在出现堆栈问题或缓存遗留问题
时才能根据问题的出发点找到问题的出处。2009-10-08
只要是引用都放在栈中
String是个特殊的类,如果形如String s = "ddd";这样形式的东西,java维护了一个常量池以提高效率,如果常量池中有,就直接把地址给s,如果没有,在常量池正宗创建一个,再把地址给s
String ss = new String("fdff");这种做法,二话不说先直接去堆中创建一个对象,然后把这个地址给ss,接着它还会去常量池中去看看有没有这个字符串,如果没有,也在常量池中创建一个。如果有就算了。怎么获取这个常量池中对应的对象呢?String提供了方法intern2009-10-08
首先java中可以分为两种数据类型:基本类型,就是那个八种,还有就是引用类型-》其他任何类型,包括数组,等
String str = new String("abc"),会在栈中分配String的引用str,指向堆中的一块对象,其中存放abc2009-10-08
基本上说的全都有问题。
首先,堆是堆,栈是栈,不能放在一起说。
Java虚拟机的操作数栈和常规意义的栈不尽相同。Java虚拟机的绝大多数运算都依赖栈,因为Java指令基本上只有常量操作数。
String a = "abc";
String b = new String("abc");
就拿这个来说,编译结果是
ldc #2 把2号常量对象("abc")的引用压栈
astore_1 把栈顶的内容弹出到1号局部变量
new #3 用3号类型(String)建立对象,并把引用压栈
dup 复制栈顶
ldc #2 和上面的ldc #2一样
invokespecial #4 调用4号方法(String(String), String的初始化方法)
astore_2 把栈顶弹出到2号局部变量
可以看到Java的局部变量不是保存在栈里的,栈的角色更像常规环境的寄存器。 (当然,对于局部变量保存空间在虚拟机内部一般是以栈模式来处理的,但这个和操作数栈是相互独立的。而且虚拟机的实现也完全可以用堆来处理局部变量,这个并不是绝对的。)2009-10-08
先放入堆内存中,取用的时候数据入栈2009-10-08