1.静态是什么?有什么用?
static的主要作用在于创建独立于具体对象的域变量或者方法。
静态代码在程序运行之前,即编译阶段,分配内存。
每创建一个对象,都会在堆里开辟内存,存成员(属性),但是不存方法,方法是共用的,没必要每一个对象都浪费内存去存方法。有一个叫方法区的地方存方法。方法区里又有静态域,存静态变量或者静态方法。
普通变量和方法的调用:通过对象调用
静态变量和方法的调用:通过对象或类调用
public class MyTest8 { public static void main(String[] args) { System.out.println(Plant.name);//通过类调用 Plant p1=new Plant(); System.out.println(p1.name);//通过对象调用 p1.name="植物";//类似方法,是公用的,改了之后就是改了,新建对象也是这个值 Plant p2=new Plant(); System.out.println(p2.name);//通过对象调用 } } class Plant{ static String name="静态植物"; public static void say() { System.out.println("植物类"); } } /**输出 静态植物 静态植物 植物 */
在类被加载的时候,静态的代码都会被加载,并且只加载一次,放在方法区里,该类的对象一起用。省时省力。
2.加载类有什么用?
任何程序都要先加载到内存中才能和CPU进行交流,而JVM中的ClassLoader(类加载器)就是负责提前将,class文件加载到内存中去的。
3.类何时被加载?(普通人的宏观理解)
- 实例化对象时,如Chinese c1=new Chinese();此时加载了Chinese类
- 通过类名调用静态变量或静态方法的时候
- 如果实例化子类对象,会先加载父类
通过代码验证:
public class MyTest8 { public static void main(String[] args) { Plant.say();//通过类名调用静态方法 Plant p1 = new Plant();//创建对象 Plant p2 = new Plant(); } } class Plant{ static String name="静态植物"; public static void say() { System.out.println("植物类"); } static { System.out.println("植物类的静态代码被执行加载了"); } { System.out.println("植物类的普通代码被加载了"); } }
输出:
植物类的静态代码被执行加载了
植物类
植物类的普通代码被加载了
植物类的普通代码被加载了
由此可知,Plant.say();调用方法时就加载了类,静态代码也都被执行了,并且只执行一次,创建对象的时候再次加载类,但是不执行静态代码,而是非静态代码。
再通过继承关系观察静态和非静态的关系
public class MyTest8 { public static void main(String[] args) { Flower.say();//通过类名调用静态方法 Flower f1=new Flower();//创建花类对象 Flower f2=new Flower(); } } class Plant{ static String name="静态植物"; public static void say() { System.out.println("植物类"); } static { System.out.println("植物类的静态代码被执行加载了"); } { System.out.println("植物类的普通代码被加载了"); } } class Flower extends Plant{ static { System.out.println("花类的普通代码被加载了"); } static String name="花"; public static void say() { System.out.println("花类"); } { System.out.println("花类的普通代码被加载了"); } }
输出:
植物类的静态代码被执行加载了
花类的普通代码被加载了
花类
植物类的普通代码被加载了
花类的普通代码被加载了
植物类的普通代码被加载了
花类的普通代码被加载了
由此可知:调用静态的东西,只执行静态的代码,静态代码只执行一次。创建对象的时候也会加载,如果静态没有被加载过也会被加载,加载过就不用。加载子类的时候,会先把父类加载一次。
在代码中,调用子类静态方法,则先加载父类的静态代码,再加载子类的静态代码,再调用方法。创建对象的时候,先加载父类,再加载子类。
类装载的过程:
- 加载:根据查找路径找到相应的 class 文件然后导入;
- 检查:检查加载的 class 文件的正确性;
- 准备:给类中的静态变量分配内存空间;
- 解析:虚拟机将常量池中的符号引用替换成直接引用的过程。符号引用就理解为一个标示,而在直接引用直接指向内存中的地址;
- 初始化:对静态变量和静态代码块执行初始化工作。
牛客刷题
1.初始化代码执行顺序是:
父类静态变量→父类静态代码块→子类静态变量→子类静态代码块→
父类普通变量→父类普通代码块→父类构造方法→
子类普通变量→子类普通代码块→子类构造方法
严格的说,静态变量+静态代码块=静态域,不包括静态方法,谁先谁后根据在类中的位置,由于一般都是先声明变量的,所以有这样的总结。这里有一个难以想象的题目,十有八九会被坑,详看代码及注释,节省篇幅。
public class B { public static B t1 = new B(); public static B t2 = new B(); { System.out.println("构造块"); } static { System.out.println("静态块"); } public static void main(String[] args) { B t = new B(); } } /** 输出: 构造块 构造块 静态块 构造块 -------------------------------------------------- 解释: 1.加载B.class 2.按照顺序先对t1和t2进行初始化,默认为null 3.又需要对t1和t2进行显示初始化,所以需要加载默认的构造方法和普通代码块 4.为什么不会加载静态代码块呢? 5.因为刚开始加载B.class时就算是加载了静态域,现在还没加载完,又不能重复加载,所以没有在初始化t1的时候输出静态块 6.t1和t2加载完了按顺序就轮到静态代码块了 7.走main方法,又创建一次B对象,执行构造方法和普通代码块 要点:静态域只加载一次,不能重复加载 */
2.静态方法属于类不属于对象;静态方法中只能访问其他静态方法和静态数据,不能访问非静态的。
3.静态变量建议别用this调用
在语法上行得通,但是静态变量是所有实例共享的一个变量,而this通常指当前对象;用对象名调用普通变量,用类名调用静态变量更加直观。很多题目直接把用this调用的情况全部判定为错。