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为成员变量。下面分析一下代码执行时候的变化:    

  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()将等待垃圾回收。

问题分析: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

 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值