一、好言
礼貌是因为陌生 胡闹是因为熟悉。
二、背景
看书,摘抄的重点,总结,不过这个大部分都是摘抄的重点,提取出来记录。《Java问题定位技术》。
三、内容
3.1 jvm内存模型
java进程内存,指整个Java进程占用的内存。即是java堆内存、Perm内存、本地内存与加载的可执行文件和库所占用内存的总和。
3.2 三个部分介绍
(1)java堆内存
Java堆内存,这是JVM 用来分配java对象内存。即通过-Xmx -Xms设置的用来分配给java对象的内存。当使用new的时候就是使用这块内存进行分配。
(2)Perm内存:其全称是Permanent Generation space ,是指内存的永久保存区域,可以通过-XX :PermSize 设置的内存,这块内存是虚拟机用来加载class字节码文件的内存。这块内存在系统运行期一般比较固定。因为类文件是有限的。这里所说的一般,还有一些不一般的情况,目前在有些面向对象的编码中,会动态进行代码织入操作,即系统在运行期间会修改或者增加字节码,这种情况下会导致类改变或者增加新的类,类需要重新加载,如果持续地有新类产生,可能会导致这块内存一直增加,直到这块内存溢出。
(3)Java进程本地内存,这是JVM用于其内部操作的内存。JVM使用的本地内存数量取决于生成的代码质量、创建的线程、GC期间用于保存java对象信息的内存,以及在代码生成、优化等过程中使用的临时空间。如果有一个第三方本地模块,那么它可能使用本地内存。例如,本地JDBC驱动程序将分配本地内存。最大本地内存量受到任何特定操作系统上的虚拟进程大小限制约束,也受到-Xmx标志指定用语java堆的内存量的限制。例如,如果应用程序能分配总计3GB的内存量,并且java堆的大小设为1GB,那么本地内存量的最大值可能2GB左右,即JVM使用的本地内存由如下几部分做成:
(a):Java.exe是C/C++写的程序,运行过程中自然需要内存,包括操作系统加载该程序,和java.exe运行过程中自己分配的内存。
(b):JNI调用动态库使用的内存,即JNI中调用new 或者malloc的内存等。
堆内存和Perm内存都可以通过参数设置它的大小,而本地内存的大小是不需要设置的,就像传统程序,内存的分配直接由操作系统从总的内存中分配。
3.3 object = null ?
把一个引用设为null的动机往往是逻辑的需要,而不是释放内存的需要。 比如:
public void clear() {
modCount++;
Entry[] tab = table;
for (int i = 0; i < tab.length; i++)
tab[i] = null;
size = 0;
}
上面代码是业务需要,并不是为了释放内存。
HashMap mapobj = new HashMap();
public void myfun(){
String obj1 = new String("abcd");
...
mapobj.put(obj1,obj1);
...
obj1 = null;
}
引用代码如下图:
实际上我们把obj1设为null,并没有回收"abcd",其还被hashmap引用,所以obj1设置为空只是将obj1的引用不指向任何内存。原本obj1的作用域就是在方法内,所以这种设置为空并不会清空其所指向的对象的物理内存。
四、内存设置:
堆:
指定初始化堆内存大小:-Xms:64m;
指定最大堆内存大小:-Xmx:1024m;
Perm:
指定初始化Perm内存大小:-XX:PermSize:64m;
指定最大Perm内存大小:-XX:MaxPermSize:1024m;
如果系统需要的堆内存确实很大,无法减少Xmx的设置,可以通过设置-Xss强行将每
个线程堆栈的尺寸设小,一旦线程堆栈过长,则自动截断,从而可以让线程堆栈占用
的内存不过渡膨胀。但这个效果往往有限的。
五、内存大小:
Java进程内存 = Java堆内存+本地内存+Perm内存。