zoukankan      html  css  js  c++  java
  • Java 内存溢出分析

    相关Java内存分配知识描述

    方法区

      保存装载的类信息
    • 类的常量池
    • 字段、方法信息
    • 方法字节码
      通常和永久(Perm)关联在一起

    Java堆

    • 和程序开发密切相关
    • 应用系统对象都保存在Java堆中
    • 所有线程共享Java堆
    • 对分代GC来说,堆也是分代的
    • GC的主要工作区间

    Java栈

    • 线程私有
    • 栈由一系列帧组成(因此Java栈也叫做帧栈)
    • 帧保存一个方法的局部变量、操作数栈、常量池指针
    • 每一次方法调用创建一个帧,并压栈

    用代码执行过程描述下虚拟机执行中内存分配情况

    public class App 
    //运行时, jvm 把 App 的信息都放入方法区 
    {
        public static void main( String[] args )
        //main 方法本身放入方法区。
        {
            Sample test1 = new Sample("小明");
             //test1是引用,所以放到栈区里, Sample是对象放在堆里面
            
            test1.sayHello("");
            
            Sample.runStatic(3, 1, 1.0, null);
            
        }
    }

    执行类的方法

    public class Sample {
        //运行时, jvm 把Sample 的信息都放入方法区
        
        private String name;
        //new Sample实例后, name 引用放入栈区里,  name 对象放入堆里
        
        public    Sample(String name){
            this .name = name;
        }
        
        //print方法本身放入 方法区里。 
        //每个线程调用 改方法,则产生一个新的【栈帧】
        public void sayHello(int  a,int b){
            //进入该方法后
            //1局部变量 reference this(本身对象的引用) 压栈 存储在该栈帧的中
            //2局部变量int  a 则压栈 存储在改栈帧的中
            //3局部变量 int b 压栈
            //4局部变量 int c
            int c = 0;
            c = a + b;
            
            //Java没有寄存器,所有参数传递使用【操作数栈】 
            //方法执行过程中操作数栈的处理过程
            //1.将数值 0压栈(操作数栈)
            //2.弹出int存放局部变量c
            //3.将局部变量a压栈
            //4.将局部变量b压栈
            //5.弹出两个变量求和,将结果压栈,此时值栈中 a,b则清除。
            //6.弹出求和结果,放与局部变量c
            //7.将局部变量c压栈入栈帧中
            
            //其他说明,c=0++;则执行过程
            //1.将数值0压入,直接执行++动作,将值返回压入c
            //所以i++ 比 i=i+1;执行速度快
            System.out.println(c);
        }
        // 方法结束清理掉sayHello 栈帧
        //所以栈空间不需要垃圾回收。
        

    //静态方法略有不一样

    举例一个递归的方法  runStatic(2)  他的栈空间和执行情况

     每次递归则会增加栈空间内存,

    所以,栈空间大小决定了方法调用的深度。

    案例证明:

    public class TestStackDeep {
        private static int count=0;
        public static void recursion(long a,long b,long c){
            long e=1,f=2,g=3,h=4,i=5,k=6,q=7,x=8,y=9,z=10;
            count++;
            recursion(a,b,c);
        }
        public static void main(String args[]){
            try{
                recursion(0L,0L,0L);
            }catch(Throwable e){
                System.out.println("deep of calling = "+count);
                e.printStackTrace();
            }
        }
    }

     当栈大小设置为128k的时候,递归调用了701 次,

     当栈大小设置为256k的时候,递归调用了1817 次,

    由此可看出,栈空间调小会影响递归调用的层级,但是分配太大,又会影响内存消耗,减少线程并发量。

    配置 -Xmx20m -Xms5m -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=d:/a.dump

    将OOM时将信息输出到dump文件中

    Vector v=new Vector();
            for(int i=0;i<25;i++)
                v.add(new byte[1*1024*1024]);

    结果堆空间超出最大空间溢出,而输出的对象的大小刚好是配置的最大内存。 19M,第20个无法分配所以报错。

     更多内存使用、GC信息可以 XX:+PrintGCDetails的输出,查看各个内存块使用详情。

     

    大抵如此,某部分的内存大到了最大值。还有其他区域,比如永久区的内存溢出

    下面列举其他一些内存溢出场景

    【情况一】:   

    Java.lang.OutOfMemoryError: Java heap space:这种是java堆内存不够,一个原因是真不够,另一个原因是程序中有死循环;   

    如果是java堆内存不够的话,可以通过调整JVM下面的配置来解决: 
      < jvm-arg>-Xms3062m < / jvm-arg> 
      < jvm-arg>-Xmx3062m < / jvm-arg> 


    【情况二】 

      java.lang.OutOfMemoryError: GC overhead limit exceeded 
      【解释】:JDK6新增错误类型,当GC为释放很小空间占用大量时间时抛出;一般是因为堆太小,导致异常的原因,没有足够的内存。 
      【解决方案】: 
      1、查看系统是否有使用大内存的代码或死循环; 
      2、通过添加JVM配置,来限制使用内存: 
      < jvm-arg>-XX:-UseGCOverheadLimit< /jvm-arg> 

    【情况三】: 

      java.lang.OutOfMemoryError: PermGen space:这种是P区内存不够,可通过调整JVM的配置: 
      < jvm-arg>-XX:MaxPermSize=128m< /jvm-arg> 
      < jvm-arg>-XXermSize=128m< /jvm-arg> 
      【注】: 
      JVM的Perm区主要用于存放Class和Meta信息的,Class在被Loader时就会被放到PermGen space,这个区域成为年老代,GC在主程序运行期间不会对年老区进行清理,默认是64M大小,当程序需要加载的对象比较多时,超过64M就会报这部分内存溢出了,需要加大内存分配,一般128m足够。 

    【情况四】: 

      java.lang.OutOfMemoryError: Direct buffer memory 
      调整-XX:MaxDirectMemorySize= 参数,如添加JVM配置: 
      < jvm-arg>-XX:MaxDirectMemorySize=128m< /jvm-arg>

     【情况五】: 

    java.lang.OutOfMemoryError: unable to create new native thread 
      【原因】:Stack空间不足以创建额外的线程,要么是创建的线程过多,要么是Stack空间确实小了。 
      【解决】:由于JVM没有提供参数设置总的stack空间大小,但可以设置单个线程栈的大小;而系统的用户空间一共是3G,除了Text/Data/BSS /MemoryMapping几个段之外,Heap和Stack空间的总量有限,是此消彼长的。因此遇到这个错误,可以通过两个途径解决:
      1.通过 -Xss启动参数减少单个线程栈大小,这样便能开更多线程(当然不能太小,太小会出现StackOverflowError); 
      2.通过-Xms -Xmx 两参数减少Heap大小,将内存让给Stack(前提是保证Heap空间够用)。 


    【情况六】: 

      java.lang.StackOverflowError 
      【原因】:这也内存溢出错误的一种,即线程栈的溢出,要么是方法调用层次过多(比如存在无限递归调用),要么是线程栈太小。 
      【解决】:优化程序设计,减少方法调用层次;调整-Xss参数增加线程栈大小。

    本文旨在简要描述分析。深度十分有限。

  • 相关阅读:
    linux内存和swap
    Linux awk sort
    redis aof和rdb区别
    STL中的map、unordered_map、hash_map
    mysql 冷热备份
    redis
    linux 几个命令
    linux erase
    group by
    现在很多技术知识点缺乏来龙去脉的介绍
  • 原文地址:https://www.cnblogs.com/javaMan/p/7103578.html
Copyright © 2011-2022 走看看