class="java" name="code">public class StaticTest { private int i; public static void main(String[] args) { //i = 1; //编译报错,不能引用非静态的变量 } }?很明显有编译错误,静态方法不能引用非静态的变量。
静态资源属于类,但是是独立于类存在的。从JVM的类加载机制的角度讲,静态资源是类初始化的时候加载的,而非静态资源是类new的时候加载的。类的初始化早于类的new,比如Class.forName(“xxx”)方法,就是初始化了一个类,但是并没有new它,只是加载这个类的静态资源罢了。所以对于静态资源来说,它是不可能知道一个类中有哪些非静态资源的;但是对于非静态资源来说就不一样了,由于它是new出来之后产生的,因此属于类的这些东西它都能认识。所以上面的几个问题答案就很明确了:
1、静态方法能不能引用非静态资源?不能,new的时候才会产生的东西,对于初始化后就存在的静态资源来说,根本不认识它。
2、静态方法里面能不能引用静态资源?可以,因为都是类初始化的时候加载的,大家相互都认识。
3、非静态方法里面能不能引用静态资源?可以,非静态方法就是实例方法,那是new之后才产生的,那么属于类的内容它都认识。
?
静态代码块public class A { private static int a = B(); static { System.out.println("Enter A.static block"); } public static void main(String[] args) { new A(); } public static int B() { System.out.println("Enter A.B()"); return 1; } }?这段代码运行结果为:
Enter A.B() Enter A.static block?由此得第一个结论:静态资源的加载顺序是严格按照静态资源的定义顺序来加载的。这和周志明老师《深入理解Java虚拟机:JVM高级特性与最佳实践》中类初始化中的说法“<clinit>()方法是由编译器自动收集类中所有类变量的赋值动作和静态语句块(static{}块)中的语句合并产生的,编译器收集的顺序是由语句在源文件中出现的顺序所决定的”是一致的。 再看一个例子:
public class A { static { c = 3; //System.out.println(c); //编译报错,不能使用,未定义 } private static int c; }?从这个例子得出第二个结论:静态代码块对于定义在它之后的静态变量,可以赋值,但是不能访问。PS:一般我们也不会这样写代码。 最后一个小例子:
public class A { static { System.out.println("A.static block"); } public A() { System.out.println("A.constructor()"); } }?
public class B extends A { static { System.out.println("B.static block"); } public B() { System.out.println("B.constructor()"); } public static void main(String[] args) { new B(); new B(); } }?执行这段代码,输出结果为:
A.static block B.static block A.constructor() B.constructor() A.constructor() B.constructor()?这个比较好理解,这个例子得出第三个结论:静态代码块是严格按照父类静态代码块->子类静态代码块的顺序加载的,且只加载一次。 ? static修饰类 static修饰的类为静态内部类,比如单例设计模式,有一种方式就可以用静态内部类来实现。 ? import static 这个比较冷门,一般用于导入静态常量或方法的包。
?
?
?
转自:http://www.cnblogs.com/xrq730/p/4820992.html
<audio controls="controls" style="display: none;"></audio>