Java堆栈存储以及Java Heap Space
一,java内存分配
1.寄存器:程序无法控制
2.栈内存:存放基本类型的数据,以及对象的引用,但对象本身存放在堆内存中(自己总结:存放基本类型的变量数据和常量以及对象的引用)
3.堆内存:存放由new创建的对象和数组
4.静态域:存放对象中用static定义的静态成员
5.常量池:存放基本常量和字符串常量
6.非RAM存储:磁盘等永久存储空间
堆和栈:
堆是由垃圾回收来负责的,可以动态分配内存大小,生存期不必告诉编译器,缺点就是存取速度慢
栈的存取速度较快,当程序运行到其作用域之外,就可以被释放,栈的数据可以被共享
字符串:
对于字符串,如果是编译器已经创建好的(String a=“qq”)就存储在常量池中,如果是运行期创建的(通过new)就存储在堆当中
成员变量和局部变量在内存中的分配
对于成员变量和局部变量:成员变量就是方法外部,类的内部定义的变量;局部变量就是方法或语句块内部定义的变量。局部变量必须初始化。 形式参数是局部变量,局部变量的数据存在于栈内存中。栈内存中的局部变量随着方法的消失而消失。 成员变量存储在堆中的对象里面,由垃圾回收器负责回收。
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 change(int i) {
i = 1234;
}
}
对于以上这段代码,date为局部变量,i,d,m,y都是形参为局部变量,day,month,year为成员变量。下面分析一下代码执行时候的变化:
- main方法开始执行:int date = 9; date局部变量,基础类型,引用和值都存在栈中。
- Test test = new Test();test为对象引用,存在栈中,对象(new Test())存在堆中。
- test.change(date); i为局部变量,引用和值存在栈中。当方法change执行完成后,i就会从栈中消失。
- BirthDate d1= new BirthDate(7,7,1970); d1为对象引用,存在栈中,对象(new BirthDate())存在堆中,其中d,m,y为局部变量存储在栈中,且它们的类型为基础类型,因此它们的数据也存储在栈中。day,month,year为成员变量,它们存储在堆中(new BirthDate()里面)。当BirthDate构造方法执行完之后,d,m,y将从栈中消失。
- main方法执行完之后,date变量,test,d1引用将从栈中消失,new Test(), new BirthDate()将等待垃圾回收。
问题分析:Java heap space
Java 内存泄漏的原因
1.内存泄漏是指无用对象持续占有内存或者无用对象得不到即使释放
2.长生命周期的对象持有段生命周期的引用就有可能发生内存泄漏
不健壮代码的特征以及解决办法
1.尽早释放无用对象的引用,好的办法就是使用临时变量的时候,让引用变量退出活动域后,自动设置为null,暗示垃圾回收来收集该对象,防止内存泄漏
2.避免使用大量的字符串处理,避免使用String,因为String对象都得独立占用内存中的一块区域,应该使用StringBudder代替
3.尽量少使用静态变量,gc不会回收
4.避免集中创建对象,尤其是大对象,jvm会突然需要大量内存,这是必然会触发gc优化系统环境,显示的声明数组空间,而且申请数量还极大
5.尽量运用对象池技术以提高系统性能,例如大集合对象拥有大量数据量的业务对象的时候,可以考虑分块处理
6.不要在经常调用的方法中创建对象,尤其是忌讳在循环中创建对象,可以适当的使用hashtable,vertor创建一组对象容器,然后从容器中去取那些对象,而不用每次new之后又丢弃
堆栈溢出一般调整的参数说明:
-Xss:单个线程的堆栈大小,jdk1.5以后每个堆栈大小为1m
-Xms:启动应用时,堆栈内存的初始值大小
-Xmx:应用运行时,jvm(java虚拟机)的极限值,为了不消耗扩大jvm堆控件分配的开销,将此参数和-xms这俩个值设置为相等,一般设置为总内存的80%
-Xmn:此参数硬性规定堆内存新生代空间大小,推介为堆内存大小的1/4
-XX:PermSize应用服务器启动时,永久存储区的初始值大小
-XX:MaxPermSize 应用服务器运行期,永久存储区的极限值大小
-XX:UserParNewGC 选择垃圾收集器为并行收集器,此配置对年轻代有效,如果你的服务器有多个CPU,开启此参数,多个CPU并发垃圾回收
-XX:ParalleGCThreads 年轻代并行垃圾收集的前提下的线程数,即:同时多少个线程一起进行垃圾回收
-Xnoclassgc 每次永久存储区满了以后,一般gc算法在做拓展分配内存前都会都会出发一次full gc,除非设置了-Xnoclassgc