zoukankan      html  css  js  c++  java
  • 理解JVM2 栈内存,方法区,堆内存

    堆,方法区,栈的关系

    分配最大堆内存-Xmx32m

    class SimpleHeap(val id: Int){
        fun show() = println("My id is $id")
    }
    
    fun main(args: Array<String>) {
        val s1 = SimpleHeap(1)
        val s2 = SimpleHeap(2)
    
        s1.show()
        s2.show()
    }

    方法区内保存类的基本信息,包括方法的实现。方法区里面的信息很少清除

    Java堆内保存着s1,s2的实例

    Java栈内保存着s1和s2的方法show()的局部变量


    栈的溢出测试

    栈帧包括:局部变量表(原生类型或引用类型的对象引用),操作数栈(类似于寄存器结构,用于计算),帧数据区(常量池指针和异常处理表)

    分配最大栈内存-Xss228K

    var count = 0
    class SimpleHeap{
        fun show(){
            count++
            val KB = ByteArray(1024*10)
            KB.set(count, count.toByte());
            show()
        }
    }

    上面代码,我以为是保存了10K的局部变量,后来发现数组还是放在堆内存里面的,栈中只保存一个引用。所以还是能递归3000次吧

    var count = 0
    class SimpleHeap{
        fun show(){
            count++
            val a = 1L
            val b = 2L
            val c = 3L
            val d = 4L
            val e = 5L
            val f = 6L
            val g = 7L
            val h = 8L
            val i = 9L
            val j = 10L
            show()
        }
    }

    一共调用670次,每次调用会使用350个字节,然后局部变量会保存80字节的long型局部变量


    堆内存回收分析

    class SimpleHeap{
        fun gc1(){
            val MB = ByteArray(1024*1024*6)
            System.gc() //不会马上回收内存
        }
    
        fun gc2(){
            var MB: ByteArray? = ByteArray(1024*1024*6)
            MB = null
            System.gc()
        }
    
        fun gc3(){
            {
                var MB = ByteArray(1024*1024*6)
            }
            System.gc()
        }
    
        fun gc4(){
            {
                var MB = ByteArray(1024*1024*6)
            }
            val c = 10
            System.gc()
        }
    
        fun gc5(){
            gc1()
            System.gc()
        }
    }

    gc1()

    这里写图片描述 
    可以看到没有回收内存。

    gc2()

    这里写图片描述 
    可以发现,又多分配了6MB,然后马上回收,这次一次性回收了12MB,因为gc1()的6MB也给回收了。

    gc3(),gc4(),gc5()

    这里写图片描述 
    gc3()gc4()不能为什么,根本没有分配内存,说不定给Kotlin编译器给优化了。 
    gc5()在gc1()退出作用域后,直接回收掉了6MB


    栈上分配内存

    对于那些线程私有的对象(指不会被其他线程访问到的对象),可以打散分配在栈上,而不是分配在堆上。在函数调用后自行下载,而不用垃圾收集器。

    实现的技术是进行逃逸分析-XX:+DoEscapeAnalysis

    /*
    -server -Xmx10m -Xms10m 
    -XX:+PrintGC -XX:+DoEscapeAnalysis 
    -XX:-UseTLAB -XX:+EliminateAllocations
    */
    class OnStackTest{
        class User(val id:Int = 0, val name:String = ""){
        }
        companion object {
            fun alloc(){
                val u = User(5,"owen")
            }
        }
    }
    
    fun main(args: Array<String>) {
        val b = System.currentTimeMillis()
        for (i in 0..1000000000){
            OnStackTest.alloc()
        }
        val c = System.currentTimeMillis()
        println(c - b)
    
    }

    要调用100000000次,按理说会频繁调用GC,但栈上分配技术显示, 咳咳,按理说是看不到GC日志

    [GC (Allocation Failure)  2047K->544K(9728K), 0.0032233 secs]
    [GC (Allocation Failure)  2592K->472K(11776K), 0.0080457 secs]
    60

    方法区

    保存系统的类信息,比如类的字段,方法,常量池。

    Java1.6,1.7可以理解为永久代(Perm),设置参数为-XX:PermSize=5m,-XX:MaxPermSize=5m

    Java1.8中变成了元数据区,使用-XX:MaxMetaspaceSize指定,这是一块堆外的直接内存,如果不指定大小,虚拟机会耗尽所有系统的可用内存

    这里写图片描述

  • 相关阅读:
    Linux 中改变主机名的 4 种方法
    如何成为优秀开发人员(一):怎样算是优秀的?
    Java中需要知道的关键字
    Java集合类常见的问题
    如何在 Linux 上复制文件/文件夹到远程系统?
    你还在 Select * 吗?
    技术人解决问题的思路
    如何创建编程语言,以及设计决策中的内容?
    Java内存溢出异常(下)
    如何在 Linux 中查看可用的网络接口
  • 原文地址:https://www.cnblogs.com/jtlgb/p/8743145.html
Copyright © 2011-2022 走看看