Java复习2.程序内存管理
前言:
国庆节的第三天,大家都回家了,一个人在宿舍好无聊。不过这年头与其说是出去玩不如是说出去挤,所以在学校里还是清闲的好。找工作不用担心了,到时候看着你们慢慢忙;插个话题,大学都没有恋爱过,总之各种原因了;大学毕业之后希望可以早点成家立业,不想一个人飘着了,所以看我笔记的人最好的是给我介绍女朋友了,PS非诚勿扰。
开始正题:之前学习了C++的内存管理,对于写程序有很大的帮助。最近在复习Java,虽然Java没有和C++那样复杂的内存操作,但是编写代码的时候还是要关注Java程序的内存管理知识。好的我们结合C/C++和Java的内存管理整理一下程序的内存管理。
1.首先看一下C++的一段代码:
char *a = "yang"; // yang是保存在常量存储区的, a只是一个指针,a的内容是指向这个字符串的首地址。
char *b = "yang";// 同样 b也是一个指针,b的内容也是一个地址,指向在常量存储区的yang首地址。
cout << a << endl; // yang
cout << b << endl; // yang //输出这两个字符串,都是yang
// 我们看一下a,b 中的内容是什么,就是一个地址,指向yang的首地址 所以输出是一样的。同理之后的判断也就当然是a==b了。
printf("a=%d\n",a); // int 地址空间
printf("b=%d\n",b); // int 地址空间
if(a==b) {
cout << "a==b" << endl; // a == b
}else{
cout << "a!=b" << endl;
}
C++的程序内存就先复习这么多,我们转入到Java
public static final String str0 = "yang";
// yang 这个字符串是保存在String Pool中的,str0也相当于一个地址,指向的是在内存中的yang的拷贝,之后所有在string pool 中的yang,如果不是new的字符串,那么访问的都是在内存中的String pool 的拷贝(如果你问我什么是String Pool,建议自己google一下);
public static void main(String[] args) {
// TODO Auto-generated method stub
final String str1 = "yang";
final String str2 = "yang";
String str4 = "yang";
String str5 = "yang";
// str1 str2 str4 str5 其实都是访问的在String Pool在内存中的拷贝的副本,也就是str1,str2, str4,str5 中保存的都是string pool 中的yang在内存中的同一个副本的地址。也就是当我们判断他们是否相等的时候,判断的是他们的指向的内存地址是不是相同,结果可想而知就是str0 == str1 == str2 == str4 ==str5
String str6 = new String("yang");//这一句就是在堆中创建一个新的字符串,其中yang任然是在String Pool中的那一个数据,但是我们在内存中(更具体的说是堆中)创建了一个新的yang的副本,我们的str6中保存的地址就是在新的副本中的地址,而不是之前的在栈中的地址。所以 直接比较他们是不相等的。
System.out.println(str0 == str1);
System.out.println(str1 == str2);
System.out.println(str4 == str5);
System.out.println(str1 == str4);
System.out.println(str0 == str6);
}
String Pool的知识好多同学都没有接触过,所以可能比较难以理解,如果真的想搞开发的话,尤其是后台开发,那么这个关于程序中的内存的知识还是十分重要的。(有点晕吧,初学的时候也是有点晕)
2.专心看一下Java程序的内存管理知识
在C++中的内存管理new/delete, malloc/free,这种方式创建的对象都是保存在堆中的,需要我们自己去管理这些内存的知识,所以会比较麻烦,其实自身感觉学习完C/C++内存管理之后,反倒是认为这种方式让我们更加明确程序的运行。而对于其他的数据存放在堆栈中的数据,当对象超出作用域的时候,就会自动销毁失效;还有一部分的数据是保存在用户存储区的,这部分的数据是在程序入口之前初初始化好的,然后再程序结束的时候才会去销毁的。
在看一下Java是如何管理内存的:首先是new 关键字,程序中使用new为每一个对象申请的内存空间(基本类型除外),所有的对象都是保存在堆中的(Heap),这些对象的释放是有GC决定和执行的,也就是我们程序员不用管这些在堆中的对象空间何时被释放,相对C++来说,省却了释放堆中对象内存的操作;Java中所有的内存释放都是GC完成的,内存的分配都是程序完成的,这样一进一出的内存管理方式简化了程序员的工作。但是这种方式加重了JVM的工作,一定程度上来说,这也造成了Java运行的速度低于C/C++的原因之一。因为GC为了能够正确释放对象的空间,GC必须监控每一个对象的运行状态,如对象的申请、引用、被引用、赋值等等操作,这些都是GC底层实现的。
监控每一个对象就是为了更加准确的、及时的释放掉对象的空间,而释放的原则就是该对象不会在被引用。
(PS:看起来好像省却了开发人员的工作,但是如果需要高性能的程序开发,还是必须掌握JVM底层的GC是如何回收内存空间的)
如何理解GC回收内存的机制?
可以这样理解:从程序开始的时候,所有的对象创建、引用、赋值等等都会看成一个有向图,从程序开始的地方,通过一些引用指向内存中的实际对象。如果一个对象在该程序的对象组成的有向图的中从根节点不可达,那么就会被GC 回收掉。
public static void main(String a[]){
Object obj1 = new Object();
Object obj2 = new Object();
obj1 = obj2;
//这个时候obj1就会被GC回收掉,而不用我们程序员亲自去释放掉这个对象在堆中的空间。
}
(图片引用其他网站上的,不是自己做的)
这种方式的释放堆中内存的精准地十分高,但是效率十分的低。
Java使用有向图的方式进行内存管理,可以消除引用循环的问题,例如有三个对象,相互引用,只要它们和根进程不可达的,那么GC也是可以回收它们的。
3.了解程序内存泄露的知识
C++中内存泄露的就是不会在使用的对象,在堆中一直存在,这样会造成内存的浪费;在Java中,对于那些不在会在使用到的对象,但是还是和有向图的根节点联通,对象无法被GC回收掉,造成程序的内存泄露。
虽然我们可以在程序中使用System.gc()回收内存空间,根据Java规范,并不一定会保障JVM执行垃圾回收。而且不同的JVM实现的gc是不同的。
看一段代码:
Vector<Object> v = new Vector<Object>();
for(int i = 0; i < 10; i++){
Object obj = new Object();
v.add(obj);
obj = null;
}
其实每一次循环中obj对象并没有被释放掉,因为我们将这些对象的引用放到Vector中了,所以对于我们删除掉obj对象,但是依然存在对堆中对象的引用,所以GC不会回收掉这个对象。
4.程序内存的4部分
程序启动的时候,有四大块内存空间:stack, heap, code, data segment
Heap segment:存储使用new声明的数据,在Java中,GC会自动回收,C++中需要程序员自己维护;
Stack Segment:存储局部变量的内存区域,当变量超出作用域的时候,就会自动释放掉该内存空间。
Code Segment:存储的是程序的函数;
Data Segment: 在程序开始的时候,静态变量存放在data segment中;
分析一段代码:
class Demo{
private int firistNum;
private int secondNum;
public static int temp = 3;
public Demo(int firstNum,int secondNum){
this.firstNum = firstNum;
this.secondNum = secondNum;
}
}
public class Test{
Public static void main(String [] args){
Demo test = new Demo(3,4);
}
}
首先在程序开始的时候,在data segment中存储的是静态变量 temp
然后进入程序中,在stack中有test, 然后就是 变量3,4进行拷贝到实参中 firstNum、secondNum,之后在Heap中有一个Demo的对象,释放掉stack中的firstNum secondNum,然后就是将Stack中的Demo对象的地址赋给stack中的test变量。
Java复习2.程序内存管理的更多相关文章
- C++复习12.程序内存管理
程序内存管理 20131006 一个程序在运行期间的内存是如何的对编写程序至关重要,之前整理的C++内存管理的知识和Java程序内存管理的知识.今天我们系统的整理一下程序的内存. 1.一个程序的内存有 ...
- java虚拟机学习-JVM内存管理:深入垃圾收集器与内存分配策略(4)
Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来. 概述: 说起垃圾收集(Garbage Collection,下文简称GC),大部分人都把这项 ...
- java虚拟机学习-JVM内存管理:深入Java内存区域与OOM(3)
概述 Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来. 对于从事C.C++程序开发的开发人员来说,在内存管理领域,他们即是拥有最高权力的皇帝又 ...
- 深入理解Java虚拟机(自动内存管理机制)
文章首发于公众号:BaronTalk 书籍真的是常读常新,古人说「书读百遍其义自见」还是很有道理的.周志明老师的这本<深入理解 Java 虚拟机>我细读了不下三遍,每一次阅读都有新的收获, ...
- zabbix3.0.4关于java服务端程序内存溢出的处理
关于java服务端程序内存溢出的处理 java服务端程序内存溢出会产生jvm.log文件,此时程序会挂掉,无法正常处理业务,需要重启服务 思路: 当存在jvm.log这个文件的时候则触发clean_j ...
- C程序内存管理
C程序的内存管理 熟悉Java语言的肯定知道,Java中内存管理是由虚拟机帮助我们完毕的,在C/C++中可不是这样,程序猿须要自己去分配和回收内存空间.本文记录了C程序可运行文件的存储结构.在内存中的 ...
- Java杂谈1——虚拟机内存管理与对象访问
1.理解JAVA虚拟机的内存管理 运行时的数据区 从java虚拟机的内存分配来看,一个java程序运行时包含了如下几个数据区: a) 程序计数寄存器(Program Counter Regis ...
- 【深入理解Java虚拟机】自动内存管理机制——垃圾回收机制
Java与C++之间有一堵有内存动态分配和垃圾收集技术所围成的"高墙",墙外面的人想进去,墙里面的人却想出来.C/C++程序员既拥有每一个对象的所有权,同时也担负着每一个对象生 ...
- 【深入理解Java虚拟机】自动内存管理机制——内存区域划分
Java与C++之间有一堵有内存动态分配和垃圾收集技术所围成的"高墙",墙外面的人想进去,墙里面的人却想出来.C/C++程序员既拥有每一个对象的所有权,同时也担负着每一个对象生 ...
随机推荐
- Hash冲突解决
hash的冲突不可避免的 1.开放地址法 开放地执法有一个公式:Hi=(H(key)+di) MOD m i=1,2,-,k(k<=m-1) 其中,m为哈希表的表长.di 是产生冲突的时候的增量 ...
- UNIX环境高级编程——线程同步之互斥量
互斥量(也称为互斥锁)出自POSIX线程标准,可以用来同步同一进程中的各个线程.当然如果一个互斥量存放在多个进程共享的某个内存区中,那么还可以通过互斥量来进行进程间的同步. 互斥量,从字面上就可以知道 ...
- JAVA之旅(十六)——String类,String常用方法,获取,判断,转换,替换,切割,子串,大小写转换,去除空格,比较
JAVA之旅(十六)--String类,String常用方法,获取,判断,转换,替换,切割,子串,大小写转换,去除空格,比较 过节耽误了几天,我们继续JAVA之旅 一.String概述 String时 ...
- 设计模式之——工厂模式(C)
本文是在学习中的总结,欢迎转载但请注明出处:http://blog.csdn.net/pistolove/article/details/41219129 主要介绍抽象工厂模式,以及工厂方法和抽象工厂 ...
- EFI怎么装系统? UEFI BIOS
关于EFI的介绍,就不赘述了. 大家可以看看这个帖子 http://benyouhui.it168.com/thread-2488583-1-1.html 总之,新电脑都是这玩意,win8也做了相应E ...
- 第三方ProgressHUD进度条 技术分享
创建一个进度辅助视图: 初始化: - (void)viewDidLoad { [super viewDidLoad]; <span style="color: rgb(255, 0, ...
- shell的case语句简述(shell的流控制)
shell流控制:http://www.cnblogs.com/yunjiaofeifei/archive/2012/06/12/2546208.html 1.if then else 语句 if t ...
- Python+Visual Studio
一直在找一个比较好的Python IDE,无奈找来找去都不太好用,由于经常用Visual Studio,所以很希望找到一个能够在VS中的Python扩展.今天发现了一个很给力的VS扩展,可以在VS中方 ...
- UILabel-UITextField-UIBotton UI_…
注意:AppDelegate是类,所以self在这个类中指的就是AppDelegate对象 - (BOOL)application:(UIApplication *)application didFi ...
- (NO.00001)iOS游戏SpeedBoy Lite成形记(二十八):增加排行榜功能
游戏大体上基本也就完成了,还差一个排行榜.否则如何激励各位选手创造新纪录呢? 排行榜功能也没什么难的,不过需要一点点排序的算法上的考虑. 这里我们把排行榜记录数据和排序都放在GameState类中,在 ...