首先说一下,Java中有哪些代码块.
普通代码块
就是在方法后面使用"{}"括起来的代码片段,不能单独执行,必须调下其方法名才可以执行.
静态代码块
在类中使用static修饰,并使用"{}"括起来的代码片段,用于静态变量的初始化或对象创建前的环境初始化.
同步代码块
使用synchronize关键字修饰,并使用"{}"括起来的代码片段.它表示在同一时间只能有一个线程进入到该方法快中,是一种多线程保护机制.
构造代码块
在类中没与任何的前缀或后缀,并使用"{}"括起来的代码片段.
简单的例子:
public class Client {
{//构造代码块
System.out.println("执行构造代码块");
}
public Client() {
System.out.println("执行无参构造函数");
}
public Client(String string) {
System.out.println("执行有参构造函数");
}
}
这是一丢按非常简单的代码,它包含了构造代码块,无参构造,有参构造.我先看一个问题,我们知道代码块不具有独立执行的能力,那么编译器是如何处理构造代码块的呢?很简单,编译器会把构造代码块插入到每个构造函数的最前端.这样 上面的代码就等同于:
public class Client {
public Client() {
System.out.println("执行构造代码块");
System.out.println("执行无参构造函数");
}
public Client(String string) {
System.out.println("执行构造代码块");
System.out.println("执行有参构造函数");
}
}
基本的理解后,我们再来看下其和构造函数的执行顺序.由于是插入到构造函数的的前面,自然在通过new关键字生成一个实例的时候会先执行构造代码块,然后在执行其他代码(注意:构造代码块不是在构造函数之前运行,而是依托于构造函数).接着我们来看一下两个主要的应用场景:
1.初始化实例变量
如果每个构造函数都需要初始化变量,即可通过构造代码块来实现.从而取代在每个构造函数调用初始化实例变量的方法.
2.初始化实例环境
一个对象必须在适当的场景下才能存在,如果没有适当的场景,则就需要在创建对象的时候创建此场景.
以上两个场景都是利用了构造代码块的两个特性:
1.在每个构造函数中都运行
2.在构造函数中它会首先运行
首先看一段代码,使用构造代码块做对象计数器.
public class Client {
public static int count = 0;
{
count++;
}
public Client() {
}
public Client(int i) {
this();
}
public Client(String string) {
}
public static void main(String[] args) {
new Client();
new Client(1);
new Client("1");
System.out.println(Client.count);
}
}
这个代码真的达到我们预期的效果吗?你可能会对this()产生了质疑.
答案是:3.
显然Java编译器是足够聪明的.这是因为,在插入到每个构造函数中的时候,有个例外,就是如果遇到this关键字(也就是构造函数调用自身其他的构造函数时)不插入构造代码块.
那为什么编译器这么聪明呢?这是因为构造代码块的出现就是为了提取构造函数的共同量,减少各个构造函数的代码而产生的.
灵活适当的使用构造代码块会让你的代码更加的简约和清晰.代码的质量自然很高很多,逼格也高了许多,有没有.
最后还有一点需要注意的,千万不要认为this是特殊情况,那super也会类似处理.其实不会,在构造代码块的处理上,super方法没有任何特殊的地方.编译器只是把构造代码块插入到super方法之后执行而已.