前言:

国庆节的第三天,大家都回家了,一个人在宿舍好无聊。不过这年头与其说是出去玩不如是说出去挤,所以在学校里还是清闲的好。找工作不用担心了,到时候看着你们慢慢忙;插个话题,大学都没有恋爱过,总之各种原因了;大学毕业之后希望可以早点成家立业,不想一个人飘着了,所以看我笔记的人最好的是给我介绍女朋友了,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.程序内存管理的更多相关文章

  1. C++复习12.程序内存管理

    程序内存管理 20131006 一个程序在运行期间的内存是如何的对编写程序至关重要,之前整理的C++内存管理的知识和Java程序内存管理的知识.今天我们系统的整理一下程序的内存. 1.一个程序的内存有 ...

  2. java虚拟机学习-JVM内存管理:深入垃圾收集器与内存分配策略(4)

    Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来. 概述: 说起垃圾收集(Garbage Collection,下文简称GC),大部分人都把这项 ...

  3. java虚拟机学习-JVM内存管理:深入Java内存区域与OOM(3)

    概述 Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来. 对于从事C.C++程序开发的开发人员来说,在内存管理领域,他们即是拥有最高权力的皇帝又 ...

  4. 深入理解Java虚拟机(自动内存管理机制)

    文章首发于公众号:BaronTalk 书籍真的是常读常新,古人说「书读百遍其义自见」还是很有道理的.周志明老师的这本<深入理解 Java 虚拟机>我细读了不下三遍,每一次阅读都有新的收获, ...

  5. zabbix3.0.4关于java服务端程序内存溢出的处理

    关于java服务端程序内存溢出的处理 java服务端程序内存溢出会产生jvm.log文件,此时程序会挂掉,无法正常处理业务,需要重启服务 思路: 当存在jvm.log这个文件的时候则触发clean_j ...

  6. C程序内存管理

    C程序的内存管理 熟悉Java语言的肯定知道,Java中内存管理是由虚拟机帮助我们完毕的,在C/C++中可不是这样,程序猿须要自己去分配和回收内存空间.本文记录了C程序可运行文件的存储结构.在内存中的 ...

  7. Java杂谈1——虚拟机内存管理与对象访问

    1.理解JAVA虚拟机的内存管理 运行时的数据区 从java虚拟机的内存分配来看,一个java程序运行时包含了如下几个数据区: a)     程序计数寄存器(Program Counter Regis ...

  8. 【深入理解Java虚拟机】自动内存管理机制——垃圾回收机制

      Java与C++之间有一堵有内存动态分配和垃圾收集技术所围成的"高墙",墙外面的人想进去,墙里面的人却想出来.C/C++程序员既拥有每一个对象的所有权,同时也担负着每一个对象生 ...

  9. 【深入理解Java虚拟机】自动内存管理机制——内存区域划分

      Java与C++之间有一堵有内存动态分配和垃圾收集技术所围成的"高墙",墙外面的人想进去,墙里面的人却想出来.C/C++程序员既拥有每一个对象的所有权,同时也担负着每一个对象生 ...

随机推荐

  1. linux中echo的用法

    1.echo命令我们常用的选项有两个,一个是-n,表示输出之后不换行,另外一个是-e,表示对于转义字符按相应的方式处理,如果不加-e那么对于转义字符会按普通字符处理. 2.echo输出时的转义字符 \ ...

  2. Android中JNI编程详解

    前几天在参加腾讯模拟考的时候,腾讯出了一道关于JNI的题,具体如下: JNI本身是一个非常复杂的知识,但是其实对于腾讯的这道题而言,如果你懂JNI,那么你可能会觉得这道题非常简单,就相当于C语言中的h ...

  3. Dynamics CRM 2013 Homepage Ribbon 按钮引用多个Javascript资源

    在CRM的开发中ribbon的开发是比较重要的一环,很多客制化的功能都需要动用ribbon区,CRM2013中的名字已经改叫command bar了,但从老版本过来的人都还是习惯叫他ribbon. R ...

  4. 初探linux子系统集之timer子系统(一)

    一般来说要让整个linux系统跑起来,那么一个必须的就是linux的时钟,也就是时间子系统了,这里正好工作需要,那么就研究下linux下的时间子系统了. linux内核必须完成两种主要的定时测量.一个 ...

  5. [C++学习历程]Visual Studio 2010 中文旗舰版 安装

    作者: 苏生米沿 本文地址:http://blog.csdn.net/sushengmiyan/article/details/19765441 要开始学习C++了,先装个开发环境吧,没有选择最新的2 ...

  6. UNIX环境高级编程——Linux系统调用列表

    以下是Linux系统调用的一个列表,包含了大部分常用系统调用和由系统调用派生出的的函数.这可能是你在互联网上所能看到的唯一一篇中文注释的Linux系统调用列表,即使是简单的字母序英文列表,能做到这么完 ...

  7. Java 8新特性探究(三)泛型的目标类型推断

    简单理解泛型 泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.通俗点将就是"类型的变量".这种类型变量可以用在类.接口和方法 ...

  8. CodePen最佳实例分享

    原文地址: Chris Coyier's Favorite CodePen Demos 原文日期: 2013年8月13日 翻译日期: 2013年8月21日 CodePen: Build, Explor ...

  9. 关于reverse_iterator

    这个reverse_iterator曾经搞得我头大,其对应的函数也是那么的可不理解...现在一切都好了. 对于left_null>1->2->3->4->right_nu ...

  10. OpenCV stereo matching BM 算法

    一直找不到opencv stereo matching的根据和原理出处,下面这个文章贴了个链接,有时间看看: Basically OpenCV provides 2 methods to calcul ...