JAVA内存的一些总结(一) 内存的划分
2013-01-31 15:01
267 查看
原博客地址 http://www.solr.cc/blog/?p=217
最近面试总是有人 信誓旦旦的问:“对java的内存了解多少?”“聊聊java的回收机制?”,”遇没有遇到过内存的问题” 巴拉巴拉的。
使用java几年了,但是自我感觉停留在应用阶段小菜鸟一般。多数开发人员应该也和我一样,除了调整一下java的内存大小以外,一般也不会遇到内存泄露的问题。除非代码写的各种资源不释放如 connection 、流等该回收的不回收。但我觉得这是代码习惯问题。既然有人面试问了,虽然打死我也不信那个面试官对内存精通,但是有人问了 我觉得我们有必要积累一下。学习一下。对于技术问题我向来是认真对待的。我只是记录自己的观点,应该具备一定的扫盲功能。^_^
我打算暂时从三方面研究内存,其余的有时间再研究:
内存的区域划分。
内存对象访问。
垃圾收集。
内存学习均来自书籍+网络+自己理解,而且是边学边写,有不对的地方忘看官指正(kongxp@yahoo.cn)
一)内存的区域划分。上图:网上找来的一个图片
这是内存运行时的一个示意图,包括线程共享(方法区、堆区)的数据区和线程独立的(栈,本地方法栈、程序计数器)数据区两大类。其实还有第三类—直接内存。
第一类:线程共享的数据区:
堆 :常说的堆内存,虚拟机管理的最大内存块。所有的对象实例及数组都在堆上开辟空间。也是垃圾回收执行的重点区域。关于垃圾回收将在以后讨论。
方法区:也叫非堆,我们经常称其为“永久代”。用于加载类信息,常量,静态变量等。虚拟机没有真正的永久代,垃圾回收也会做些工作,但是对我们比较透明。知道为什么public static final 是干啥的了吧!他在方法区中存放,而且理论上不被回收。
第二类:线程独立的数据区
程序计数器:内存占用小,主要完成字节码的行号指示器。用于选中下一条执行指令。
栈:常说的内存栈。生命周期同线程。用于存放局部变量表。你听这名字,局部变量表。结合java的常识不难推测出,它存储的就是基本数据类型,对象引用(就是地址)和字节码指令地址(罕见,其实就是returnaddress)。值得注意的是double和long为64位需要两个局部变量空间,而其他的一切数据类型只占有一个局部变量空间。
本地方法栈:native方法使用的栈。
第三类:直接内存:不是java虚拟机运行时的数据区。有些native方法可以直接开辟堆外内存,提高性能。有那些,我没研究过。找到了在更新这部分。
所以以后再聊内存时,我们不能简单说堆和栈两类。应该分为共享区和独立(隔离)区。又分为堆 方法区 栈 本地方法区 程序计数器五部分。
附:上文中提及的native方法的说明
Java不是完美的,Java的不足除了体现在运行速度上要比传统的C++慢许多之外,Java无法直接访问到操作系统底层(如系统硬件等),为此Java使用native方法来扩展Java程序的功能。
可以将native方法比作Java程序同C程序的接口,其实现步骤:
1、在Java中声明native()方法,然后编译;
2、用javah产生一个.h文件;
3、写一个.cpp文件实现native导出方法,其中需要包含第二步产生的.h文件(注意其中又包含了JDK带的jni.h文件);
4、将第三步的.cpp文件编译成动态链接库文件;
5、在Java中用System.loadLibrary()方法加载第四步产生的动态链接库文件,这个native()方法就可以在Java中被访问了。
具体的办法实现网上例子很多。
最近面试总是有人 信誓旦旦的问:“对java的内存了解多少?”“聊聊java的回收机制?”,”遇没有遇到过内存的问题” 巴拉巴拉的。
使用java几年了,但是自我感觉停留在应用阶段小菜鸟一般。多数开发人员应该也和我一样,除了调整一下java的内存大小以外,一般也不会遇到内存泄露的问题。除非代码写的各种资源不释放如 connection 、流等该回收的不回收。但我觉得这是代码习惯问题。既然有人面试问了,虽然打死我也不信那个面试官对内存精通,但是有人问了 我觉得我们有必要积累一下。学习一下。对于技术问题我向来是认真对待的。我只是记录自己的观点,应该具备一定的扫盲功能。^_^
我打算暂时从三方面研究内存,其余的有时间再研究:
内存的区域划分。
内存对象访问。
垃圾收集。
内存学习均来自书籍+网络+自己理解,而且是边学边写,有不对的地方忘看官指正(kongxp@yahoo.cn)
一)内存的区域划分。上图:网上找来的一个图片
这是内存运行时的一个示意图,包括线程共享(方法区、堆区)的数据区和线程独立的(栈,本地方法栈、程序计数器)数据区两大类。其实还有第三类—直接内存。
第一类:线程共享的数据区:
堆 :常说的堆内存,虚拟机管理的最大内存块。所有的对象实例及数组都在堆上开辟空间。也是垃圾回收执行的重点区域。关于垃圾回收将在以后讨论。
方法区:也叫非堆,我们经常称其为“永久代”。用于加载类信息,常量,静态变量等。虚拟机没有真正的永久代,垃圾回收也会做些工作,但是对我们比较透明。知道为什么public static final 是干啥的了吧!他在方法区中存放,而且理论上不被回收。
第二类:线程独立的数据区
程序计数器:内存占用小,主要完成字节码的行号指示器。用于选中下一条执行指令。
栈:常说的内存栈。生命周期同线程。用于存放局部变量表。你听这名字,局部变量表。结合java的常识不难推测出,它存储的就是基本数据类型,对象引用(就是地址)和字节码指令地址(罕见,其实就是returnaddress)。值得注意的是double和long为64位需要两个局部变量空间,而其他的一切数据类型只占有一个局部变量空间。
本地方法栈:native方法使用的栈。
第三类:直接内存:不是java虚拟机运行时的数据区。有些native方法可以直接开辟堆外内存,提高性能。有那些,我没研究过。找到了在更新这部分。
所以以后再聊内存时,我们不能简单说堆和栈两类。应该分为共享区和独立(隔离)区。又分为堆 方法区 栈 本地方法区 程序计数器五部分。
附:上文中提及的native方法的说明
Java不是完美的,Java的不足除了体现在运行速度上要比传统的C++慢许多之外,Java无法直接访问到操作系统底层(如系统硬件等),为此Java使用native方法来扩展Java程序的功能。
可以将native方法比作Java程序同C程序的接口,其实现步骤:
1、在Java中声明native()方法,然后编译;
2、用javah产生一个.h文件;
3、写一个.cpp文件实现native导出方法,其中需要包含第二步产生的.h文件(注意其中又包含了JDK带的jni.h文件);
4、将第三步的.cpp文件编译成动态链接库文件;
5、在Java中用System.loadLibrary()方法加载第四步产生的动态链接库文件,这个native()方法就可以在Java中被访问了。
具体的办法实现网上例子很多。
相关文章推荐
- Java内存布局和类加载的一些总结
- 关于java对象内存的一些总结
- java内存区域划分及原理。网上总结。
- JAVA内存的一些总结(二) 对象访问
- JAVA内存的一些总结(三) 垃圾回收
- 对java内存回收机制的一些总结
- 总结一些java中的基础知识
- 牛客网Java刷题知识点之内存的划分(寄存器、本地方法区、方法区、栈内存和堆内存)
- 《深入理解 Java 内存模型》笔记总结
- 【Java并发编程】:深入Java内存模型—内存操作规则总结
- C/C++中程序内存区域划分大总结
- JVM 虚拟机 (一) Java内存分配 划分与回收
- JAVA将其他格式转成FLV视频的一些总结
- c语言第二周总结-程序存储区划分/动态申请内存/内存操作函数
- c/c++中内存区域划分大总结
- 总结的一些Java公用函数库
- 关于JAVA垃圾收集器与类的finalize()方法的一些总结
- 总结的一些Java公用函数库
- [Java面试六]SpringMVC总结以及在面试中的一些问题.
- 黑马程序员——java中file类的一些总结