一.栈内存里的引用变量并未真正存储对象的成员变量,对象的成员变量数据实际存放在堆内存中,而引用变量只是指向该堆内存里的对象。
二.堆内存里的对象可以有多个引用,若果堆内存中没有变量指向该对象,程序无法再次访问该对象,该对象就变为垃圾等待回收。
三. 普通代码块:在方法或语句中出现的{}就称为普通代码块。普通代码块和一般的语句执行顺序由他们在代码中出现的次序决定--“先出现先执行”
构造块:直接在类中定义且没有加static关键字的代码块称为{}构造代码块。 构造代码块在创建对象时被调用,每次创建对象都会被调用,并且构造代码块的执行次序优先于类构造函数。
//静态代码块:在java中使用static关键字声明的代码块。静态块用于初始化类,为类的属性初始化。
//每个静态代码块只会执行一次。由于JVM在加载类时会执行静态代码块,所以静态代码块先于主方法执行。
//如果类中包含多个静态代码块,那么将按照"先定义的代码先执行,后定义的代码后执行"。
//注意:1 静态代码块不能存在于任何方法体内。2 静态代码块不能直接访问静态实例变量和实例方法,需要通过类的实例对象来访问。
// 系统在类初始化阶段执行静态初始化块时,不仅会执行本类的静态初始化块,还会一直追溯到其父类的静态初始化块……最后才执行本类的静态初始化块。
对象初始化阶段,会执行最顶端的父类初始化块,对顶端的父类构造器,然后依次向下,直到执行当前类的初始化块,当前类的构造器。
class Root { static { System.out.println("Root静态初始化块"); } { System.out.println("Root普通初始化块"); } public Root() { System.out.println("Root无参构造器"); } } class Mid extends Root { static { System.out.println("Mid静态初始化块"); } { System.out.println("Mid普通初始化块"); } public Mid() { System.out.println("Mid无参构造器"); } } class Leaf extends Mid { static { System.out.println("Leaf静态初始化块"); } { System.out.println("Leaf普通初始化块"); } public Leaf() { System.out.println("Leaf无参构造器"); } }
如果初始化Leaf 对象:输出结果为
Root静态初始化块 - Mid静态初始化块-Leaf静态初始化块 - Root普通初始化块 -Root无参构造器 - Mid普通初始化块-Mid无参构造器--Leaf普通初始化块-Leaf无参构造器。如果在初始化依次Leaf对象将不再执行父类与本身的静态代码块,输出结果为上述去掉静态代码块部分。
四. 方法参数传递机制:值传递,引用传递 看堆栈分配区分
五.java1.5 形参个数可变 在方法里最后一个形参的类型加上三个点(...)表明该形参可以接受多个参数值。public void test(String...test);
六.初始化对象时会先给类变量(类)分配内存空间(分配到堆内存中)。不过也可以通过P1 和P2 调用类变量,但是本质还是通过Person类即通过Person类的内存访问的。为了避免歧义尽量不要用对象调用类变量。
七。局部变量不属于任何实例和类,因此它总是保存在其所在方法的栈内存中,如果局部变量时基本类型的变量,则直接把这个变量的值保存在该变量对应的内存中。
如果局部变量时引用类型的变量,则这个变量里存放的是地址,通过该地址引用到该变量实际引用的对象或数组。栈内存中的变量无需系统垃圾回收,往往随着方法或代码块的运行结束而结束。
八.封装的目的:隐藏类的实现细节;可进行数据检查,从而有利保证对象信息的完整性;便于修改,提高代码可维护性;将对象的成员变量和实现隐藏起来,不允许外界直接访问;把方法暴露出来,让方法来控制对这些成员变量进行安全的访问和操作。
private :同一个类中
default :同一个类中;同一个包中
protected:同一个类中,同一个包中,子类中
public:同一个类中,同一个包中,子类中,全局范围内;
九. 继承
当程序创建一个子类对象时,系统不仅会为该类中定义的实例变量分配内存,也会为他从父类继承得到的所有是咧变量分配空间。
子类构造器执行体重既没有super调用,也没有this调用,系统会在执行子类构造器之前,隐式调用父类无参的构造器。
创造任何对象总是从该类所在继承树最顶层类的构造器开始执行。然后依次向下执行,最后才执行本类的构造器
十.多肽
引用变量在编译阶段只能调用器编译时类型锁具有的方法,但运行是则执行它运行时类型所具有的方法。eg:Object c = new Person(); 只能调用object类的方法
但是 运行时的方法 是子类的方法 表现为多肽
通过引用变量来访问其包含的实例变量时,系统总是试图访问它编译时类型所定义的成员变量,而不是它运行时所定义的成员变量。
eg:父类有个字段a=1 子类有个相同名字的字段a=2 . Base b = new Sub(); System.out.println(b.a); 输出结果为 1