敞开生长之旅!这是我参加「日新计划 2 月更文挑战」的第 7 天,点击查看活动概况
static作为java中基础常用的关键字,通常用于润饰内部类,办法和变量和代码段,且具有以下特性:
- static润饰内部类时,该类归于静态内部类,其只能访问外部的静态变量和办法
- static润饰办法时,该办法归于静态办法,能够直接经过类名调用,而不用创立类实例
- static润饰变量时,该变量归于静态变量,存储于常量池中
- static润饰代码块,则该代码块会在类加载时初始化,且代码块中的逻辑多线程安全
静态内部类不会持有外部类的引证
在final那些事儿一文中,提到我们能够使用静态内部类来处理匿名Handler导致的内存走漏问题,其根本原因在于静态内部类不会持有外部类引证,测试代码如下:
public class FatherClass {
public FatherClass() {
Inner inner = new Inner();
}
public static class Inner{
}
}
查看FatherClass$Inner的字节码如下图所示:
能够看到在Inner结构函数内确实没有持有外部类FatherClass的引证,有朋友要问了,那么Inner类中引证外部类成员变量怎么办?编写测试代码如下:
能够看出当静态内部类引证外部类成员变量时,该变量有必要是静态变量,将mAge修正为static润饰后,查看FatherClass类字节码,能够看出mAge访问标志为static且存储在常量池中。
父类的static办法能够直接被子类承继吗?
修正FatherClass代码如下所示:
public class FatherClass {
private static int mAge = 100;
public FatherClass() {
Inner inner = new Inner();
}
public static void sayHello() {
System.out.println("sayHello in FatherClass");
}
public static class Inner{
public void increaseAge() {
mAge = 1000;
}
}
}
其子类代码如下所示:
public class ChildClass extends FatherClass{}
在Main类中创立FatherClass和ChildClass目标,并调用,代码如下:
public class Main {
public static void main(String[] args) {
ChildClass.sayHello();
FatherClass.sayHello();
ChildClass childClass = new ChildClass();
childClass.sayHello();
FatherClass fatherClass = new ChildClass();
fatherClass.sayHello();
}
}
履行结果如下 :
能够看出经过ChildClass的类名或实例目标均能够访问到父类的静态办法,即子类能够承继父类的静态办法。
子类能重写父类的static办法吗?
在ChildClass中使用编辑器代码自动生成功用,去挑选override父类办法,能够看出没有sayHello办法能够挑选,能够看出子类是不能重写父类的静态办法,如下图所示:
子类能定义与父类静态办法同名的静态办法吗?
修正ChildClass代码如下所示:
public class ChildClass extends FatherClass{
public static void sayHello() {
System.out.println("sayHello in ChildClass");
}
}
履行Main中的测试代码,结果如下:
能够看出,子类能够定寄父类同名的静态办法,此刻经过子类目标和子类调用的是子类声明的静态办法,当子类目标强转成父类目标或许经过父类,父类目标调用的是父类声明的静态办法。
总结