代码块

定义:在定义属性的位置上,在任何方法之外,定义一个代码块

种类:分为两类即动态代码块、静态代码块

<1> 动态初始代码块:在初始化属性之前调用初始化代码块{……}
<2> 静态初始代码块:在类加载时运行static{……}只被运行一次,往往用作一个类的准备工作

示例代码:

package mark.zhang; public class Linux { /** * 动态代码块 */ { System.out.println("--动态初始化代码块--"); } /** * 静态代码块 */ static { System.out.println("--静态初始化代码块--"); } /** * 测试 * * @param args */ public static void main(String[] args) { Linux linux = new Linux(); } }

执行结果:

--静态初始化代码块-- --动态初始化代码块--

从结果看出,创建Linux类的实例时加载这个类,从而代码块执行,只是静态代码块先于动态代码块之前执行。那么,我们思考两个问题,类何时被加载,代码块、构造方法以及成员变量之间的执行顺序是谁先谁后??汗,别急,往下看。

先说一说类加载时机吧!!

(1)new 一个对象的时候,加载

(2)没有创建对象,访问类中静态成员(方法和属性),加载

(3)声明一个类的引用,不加载

(4)创建子类,先加载父类,再加载子类

(5)父类中的公开静态方法,子类继承,使用子类的类名调用此方法,加载父类

注意,这里需要满足两个条件:

<a> 子类不可以重写该静态方法,其实父类的静态方法子类是不可以重写只可以继承的。即使重写也要加上static关键字。那么也就说明父类的静态方法是不可以被子类重写的,子类重写的这个静态方法称之为子类自己的静态方法(只不过和父类的该静态方法重名罢了!!)

<b> main方法不可以在子类中,看例子代码,如下:

package mark.zhang; public class Linux { /** * 动态代码块 */ { System.out.println("--父类Linux--动态初始化代码块--"); } /** * 静态代码块 */ static { System.out.println("--父类Linux--静态初始化代码块--"); } static void get() { System.out.println("--父类Linux--get()"); } } class Ubuntu extends Linux { /** * 动态代码块 */ { System.out.println("--子类Linux--动态初始化代码块--"); } /** * 静态代码块 */ static { System.out.println("--子类ubuntu静态初始化代码块--"); } } class Test { /** * 测试 * * @param args */ public static void main(String[] args) { Ubuntu.get(); } }

在主函数中运行以下代码:
Ubuntu.m(); //加载了父类之后,虚拟机已经知道m()方法的调用了,就不会再加载子类,延迟加载

运行结果:

--父类Linux--静态初始化代码块-- --父类Linux--get()
(6)没有创建对象,访问类中静态常量(能计算出结果的常量,在编译的时候会用计算出来的结果替换表达式),不加载

(7)没有创建对象,访问类中静态常量(不确定的值),加载

关于(6)(7),举两个个小例子吧??!!!

package my.test; public class FinalTest { // 编译时常量,编译器在编译时会将所有用到该变量的地方替换成相应的字面值 public static final int a = 4+4; static { System.out.println("--static code--"); } } class Test { public static void main(String[] args) { System.out.println("FinalTest.a= " + FinalTest.a); } }

该例子中没有执行静态代码块,正如(6)所说。

运行结果:

FinalTest.a= 8

再看一个例子,来说明(7)的正确性!!!

package my.test; public class FinalTest { // 运行时常量,在运行的时候才能确定 public static final int a = 4 + Math.abs(1); static { System.out.println("--static code--"); } } class Test { public static void main(String[] args) { System.out.println("FinalTest.a= " + FinalTest.a); } }

运行结果,执行静态代码块内容。

--static code-- FinalTest.a= 5

ok,上面说的都是关于类的加载时机问题,接着说说代码块、构造方法以及成员变量之间的执行顺序的问题。要想说明白这个问题,需要从两个方面说,一个是单纯的一个类,另一个就是一个类与另一个类的继承。

<1>单纯的一个类

package mark.zhang; public class Linux { // 静态成员变量 public static String name = "Linux OS"; // 一般变量 public int size = 2; public Linux() { System.out.println("--父类Linux--构造方法--"); } /** * 动态代码块 */ { System.out.println("--父类Linux--动态初始化代码块--"); } /** * 静态代码块 */ static { System.out.println("--父类Linux--静态初始化代码块--"); } }

测试一下,呵呵!!

class Test { /** * 测试 * * @param args */ public static void main(String[] args) { new Linux(); } }

测试结果,如下:

--父类Linux--静态初始化代码块-- --父类Linux--动态初始化代码块-- --父类Linux--构造方法--

<2> 具有继承关系的两个类

package mark.zhang; public class Linux { // 静态成员变量 public static String name = "Linux OS"; // 一般变量 public int size = 2; public Linux() { System.out.println("--父类Linux--构造方法--"); } /** * 动态代码块 */ { System.out.println("--父类Linux--动态初始化代码块--"); } /** * 静态代码块 */ static { System.out.println("--父类Linux--静态初始化代码块--"); } } class Ubuntu extends Linux { public Ubuntu() { System.out.println("--子类Ubuntu--构造方法--"); } /** * 动态代码块 */ { System.out.println("--子类Ubuntu--动态初始化代码块--"); } /** * 静态代码块 */ static { System.out.println("--子类ubuntu静态初始化代码块--"); } } class Test { /** * 测试 * * @param args */ public static void main(String[] args) { new Ubuntu(); } }

执行结果,如下:

--父类Linux--静态初始化代码块-- --子类ubuntu静态初始化代码块-- --父类Linux--动态初始化代码块-- --父类Linux--构造方法-- --子类Ubuntu--动态初始化代码块-- --子类Ubuntu--构造方法--

至于,代码块与成员变量之间谁先谁后,其实是与它们在类中声明顺序有关的。现在以一个小例子说明问题:

package mark.zhang; public class Linux {//静态与非静态成员变量与静态代码块、非静态代码块的执行顺序与其声明顺序有关 RedHat rh1 = new RedHat("非静态成员变量rh1"); static RedHat rh11 = new RedHat("静态成员变量rh11"); public Linux() { System.out.println("--父类Linux--构造方法--"); } /** * 动态代码块 */ { System.out.println("--父类Linux--动态初始化代码块--"); } /** * 静态代码块 */ static { System.out.println("--父类Linux--静态初始化代码块--"); } RedHat rh2 = new RedHat("非静态成员变量rh2"); static RedHat rh22 = new RedHat("静态成员变量rh22"); } class RedHat { static int times = 1; public RedHat(String info) { System.out.println(info + ", the order " + (times++)); } }

测试:

class Test { /** * 测试 * * @param args */ public static void main(String[] args) { new Linux(); } }

运行结果:

静态成员变量rh11, the order 1 --父类Linux--静态初始化代码块-- 静态成员变量rh22, the order 2 非静态成员变量rh1, the order 3 --父类Linux--动态初始化代码块-- 非静态成员变量rh2, the order 4 --父类Linux--构造方法--



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值