疯狂JAVA16课之对象与内存控制
- java内存管理分为两个方面:内存分配和内存回收。这里的内存分配特指创建java对象时JVM为该对象在对内存中所分配的内存空间。内存回收指的是当该java对象失去引用,变成垃圾时,JVM的垃圾回收机制自动清理该对象,并回收该对象所占用的内存。由于JVM内置了垃圾回收机制会收失去引用的java对象所占用的内存,所以很多java开发者认为java不存在内存泄漏,资源泄露的问题。实际上是一种错觉,java程序依然会有内存泄漏。
- 由于JVM的垃圾回收机制由一条后台线程完成,本身也是非常消耗性能的,因此如果肆无忌惮的创建对象,让系统分配内存,拿这些分配的内存酱油垃圾回收机制进行回收,这样做有两个坏处:
-
- 不断分配内存使得系统中可用内存减少,从而降低程序运行性能
- 大量已经分配内存的会收拾的垃圾回收的负担加重,降低程序运行性能
- 代码:
运行结果:
分析:
类变量初始化分成两个阶段:
1.系统为price的两个变量分配内存空间;
2.按初始化代码的排列顺序对类变量执行初始化
1 package section2;
2 class Base{
3 private int i =2;
4 public Base(){
5 this.display();
6 }
7 public void display(){
8 System.out.println(i);
9 }
10 }
11 class Derived extends Base{
12 private int i =22;
13 public Derived(){
14 i =222;
15 }
16 public void display(){
17 System.out.println(this.i);
18 }
19 }
20 public class Test {
21 public static void main(String[] args) {
22 new Derived(); //此时执行先执行了父类的构造方法,输出的i是本类的,
23 // 此时i还未初始化,所以为0,等输出后i才被赋值为222
24
25 }
26 }
输出0;
- 对于父类定义的public成员变量,系统将其保留在父类中,并不会将他转移到其子类中,所以父类和子类可以同时拥有同名的实例变量
因为继承成员变量和继承方法之间存在这样的差别,所以对于一个应用类型的变量而言,当通过该变量访问他所引用的对象的方法时,该实例变量的值取决于声明该变量的对象类型,当通过该变量来调用它所引用的对象的方法时,该方法行为取决于他所实际引用的对象的类型。
父 ,子类对象在内存中存储的结论:当程序创建一个子类对象时,系统不仅会为该类中定义的实例变量分配内存,也会为父类中定义的所有实例变量分配内存,即使子类定义了与父类中同名的实例变量(但父类的成员变量不属于子类,如果父类的变量为共有的可以通过子类.变量名修改父类的变量。而不是说明子类拥有了父类的变量)。也就是说,当系统创建一个java对象的时候,如果该java类有两个父类(一个直接父类A,一个间接父类B),假设A类中定义了2个实例变量,B类中定义了3个实例变量,当前类中定义了2个实例变量,那这个对象将会保存2+3+2个实例变量。(注意保存不代表拥有)
- 定义final变量时,只有在定义时指定初始值,系统才会执行”宏替换“ 。
- 如果程序需要在局部内部类(只有局部内部类才可以访问局部变量:即放在方法里定义的内部类)中使用局部变量,那么这个局部变量必须使用final修饰符修饰。因为:对于普通局部变量而言,他的作用于就是停留在该方法内,当方法执行结束,该局部变量也随之消失;但内部类则可能产生隐式的”闭包“,闭包将使得局部变量脱离它所在的方法继续存在。