在J2SE中,通过进行内存分析,可以让我们更好的理解我们的程序在内存中是怎么被分配内存的。也能让我们更好的理解我们的代码是怎么运行的。
对于我自己来说分析内存也是一件很有趣的事情。所以下面通过一个例子来总结一下程序运行过程的中内存分配.
### 程序实例
~~~
public class Point {
double x,y,z;
Point(double _x,double _y,double _z)
{
x=_x;
y=_y;
z=_z;
}
void setX(double _x)
{
x=_x;
}
double getDistance(Point p)
{return (x-p.x)*(x-p.x)+(y-p.y)*(y-p.y)+(z-p.z)*(z-p.z);}
}
~~~
~~~
public class TestPoint {
public static void main(String[] args) {
Point p=new Point(1.0,2.0,3.0);//第一步
Point p1=new Point(0.0,0.0,0.0);//第二步
System.out.println(p.getDistance(p1));//第三步
p.setX(5.0);//第四步
System.out.println(p.getDistance(new Point(1.0,1.0,1.0)));//第五步
}
}
~~~
### 内存分析
**第一步**
首先定义了一个变量,所以在栈内存中会分配一块空间来存储变量,变量名为P,它指向了一个new出来的对象,这个new出来的对象,当然是放到堆内存。
在执行的过程中调用了Point的构造方法.在执行构造方法的过程中,在栈内存中临时分配了三个空间分别存储_x,_y,_z,然后再把这三个值赋值给点对象中的x,y,z.
![](https://box.kancloud.cn/2016-03-23_56f1fee63350c.jpg)
赋值完成后这些临时分配的变量将会被取消.所以第一步执行完后的内存分配如下图
![](https://box.kancloud.cn/2016-03-23_56f1fee647396.jpg)
**第二步**
第二步的执行过程和第一步类似,不再做赘述,该步执行完后的内存分配如下图.
![](https://box.kancloud.cn/2016-03-23_56f1fee65be6a.jpg)
**第三步**
该步调用了一个方法,从方法中我们可以看到有一个形参.所以首先在栈内存中临时分配一块空间,保存这个形参,也就是局部变量P.再看getDistance()方法中我们实际传入的是P1,所以将p引用P1引用的对象(这里注意区分此刻分配的局部变量P和第一步new出来的P不是一回事)
该步的方法是有返回值的,所以需要在栈中分配空间来保存这个返回值.内存分配图如下
![](https://box.kancloud.cn/2016-03-23_56f1fee66fad7.jpg)
将返回值打印后,临时分配的内存空间取消.其内存分配图和第二步执行完后的内存分配图是相同的.
**第四步**
从调用的方法上看,首先要在栈中分配一个临时变量_x,值为5.0.然后再将值赋值给P指向的对象.执行完后临时变量被取消.内存分配图如下.
![](https://box.kancloud.cn/2016-03-23_56f1fee681766.jpg)
**第五步**
此表达式比较复杂,当一个表达式复杂的时候,我们要从里向外分析.
基于此原则,分析得到,首先要在堆内存中new一个对象,这时这个对象并没有被引用
然后执行方法.上面我们已经分析过了,执行这个方法我们需要先在栈中临时分配一个变量P,此时需要将new出的对象给临时变量P,也就是说这个P引用我们刚刚new出的那个对象。
然后计算两点的距离,这时需要一个临时变量来保存返回来的数据.
方法执行完后,所有临时分配的内存被取消.这时要注意的是我们new出来的对象是通过垃圾回收机制取消的。
内存分析如下图
![](https://box.kancloud.cn/2016-03-23_56f1fee696d3f.jpg)
### 程序执行完后的内存分配
最后展示一张程序执行完后的内存分配图
![](https://box.kancloud.cn/2016-03-23_56f1fee6b550c.jpg)
### 总结
对于内存分析之前认为挺难的,但是经过仔细分析后发现其实内存分析很简单,只要大家能够看明白以下这几点,我想对于一些简单的内存分析你一定没有问题了.
(1)形参要临时分配空间,程序执行完后要 取消空间分配.
(2)对于复杂的程序要从里到外分析.一步步来.
(3)有返回值的方法,其返回值也是要分配给一个临时变量的.同样的使用完后要取消.
(4)临时new出来的对象是利用垃圾回收机制取消的.
(5)静态成员变量时放在哪里的,是怎么分配内存的.(这个例子并没有展示静态成员变量的分配)