JVM的垃圾回收机制是由一条后台线程执行的,其本身也是非常消耗内存的,因此,滥用创建对象,会导致性能大大下降,对内存的分配的了解就显得尤为重要
变量分类
局部变量
- 形参:存在于方法签名中定义的局部变量,有方法调用者为其赋值,随着方法的结束而消亡
- 方法内部变量:在方法内部定义的局部变量,必须在方法内对其进行显示初始化,随着方法的结束而消亡
- 代码块内部的局部变量:在代码块内定义的局部变量,必须在代码块内对其显式初始化,随着代码块结束而消亡
成员变量
- 实例变量:非静态的成员变量,随着对象的产生,进行初始化等操作,对象结束变量也就消亡
- 类变量:静态的成员变量,带有static修饰符,随着类初始化产生,随着类消失而消失
在创建变量的时候,一定要合法的前向引用。其含义就是先定义的变量不能引用后定义的变量,反之则可以
变量的内存分配
在同一个JVM中每一个类只会存在一个Class对象,因此JVM只要分配一块内存空间给类变量就可以了,而实例变量则每次创建对象都要为其分配一块内存,几个实例就要创建几块内存空间
实例变量的初始化时机
- 定义变量的时候
- 代码块中
- 构造器中
定义实例变量时指定的初始值、初始化块中为实例变量指定初始值的语句的地位是平等的,当经过编译器处理后,他们都将被提取到构造器中,也就是说在编译后,初始化都会被放在构造器中按先后顺序进行初始化赋值
类变量初始化时机
同一个JVM中,类变量只能初始化一次
- 定义变量的时候
- 静态代码块
父类构造器
在创建Java对象的时候,都会先去执行该类的父类对象的非静态代码块和构造器,最后才是该类的非静态代码块和构造器
- 所谓的隐式调用和显式调用,其实就是有没有用super去调用父类的构造器的区别。
- 如果父类还没被初始化过,则会最先对类变量进行初始化
访问子类对象的实例变量
子类的方法可以访问父类的实例变量,这是因为子类继承父类就会获得父类的成员变量和方法,但父类的方法不能访问子类的实例变量,因为父类无法知道哪个子类继承他
而且子父类中的成员变量(类变量和实例变量)是相互独立的,父类中的成员变量不会被子类中同名的变量覆盖
1 | class Base { |
结果
200
20
2
当创建Sub的时候,会初始化Base、Mid和Sub三个对象,同时也就存在三个count变量了,也就是说有三块内存保存着这三个对象和count变量,以s2m变量为例,s2m拥有的地址是Sub对象的堆地址,但s2m变量类是是Mid,则会去寻找Mid下的变量值
也就说成员变量的值取决于声明该变量声明时是所用的类型
访问子类对象的方法
子类可以重写父类的方法,子类也可以通过super的方式调用父类的方法,在多态的情况下,子类重写的方法会覆盖掉父类的方法
1 | class Base{ |
结果
2
2
20
20
2
20
2
一切在你执行这段代码之后,你就会明白一切了。