1:代码块的基本介绍和基本语法
- 代码块又被称为初始块,属于类中的成员,类似于方法把逻辑语句放在{}中,但是没有名字返回值,没有参数,也不能被调用,在加载类或者创建对象的时候被隐式调用。
- 语法:
[修饰符] {
代码
};
- 修饰符可选且只能是static
- 代码块分两种一种是静态代码块,一种是实例代码块/普通代码/非静态代码块
- ;可以写上也可以省略
2:体会代码块的用处
- 相当于另外一种形式的构造体,可以进行初始化此操作
- 当多个构造器中都有重复的语句,就可以抽取到初始代码块中,提高代码重用性
class CodeBlock01 {
public static void main(String[] args) {
Movie movie = new Movie("你好,李焕英");
System.out.println("===============");
Movie movie2 = new Movie("唐探 3", 100, "陈思诚");
}
}
class Movie {
private String name;
private double price;
private String director;
{
System.out.println("电影屏幕打开...");
System.out.println("广告开始...");
System.out.println("电影正是开始...");
};
public Movie(String name) {
System.out.println("Movie(String name) 被调用...");
this.name = name;
}
public Movie(String name, double price) {
this.name = name;
this.price = price;
}
public Movie(String name, double price, String director) {
System.out.println("Movie(String name, double price, String director) 被调用...");
this.name = name;
this.price = price;
this.director = director;
}
}
3:代码块的细节
- static代码块也叫静态代码块,作用是对类进行初始化,而且它之随着类的加载而执行,且只会执行一次,如果是普通代块,每创建一个对象,执行一次
- 类加载的几种情况:创建对象实例的时候,创建子对象的实例,父类类也会被加载,使用静态成员时(静态属性,静态方法)
- 普通代码块:在创建对象实例的时候,会被隐形调用,被创建一次就会调用一次,如果使用类的静态成员,普通代码块并不会执行。
- static的=代码是类加载的时候,且只会被执行一次,普通代码块是创建对象的时候,创建一次对象执行一次
public class Code {
public static void main(String[] args) {
System.out.println(DD.n1);
}
}
class DD {
public static int n1 = 8888;
static {
System.out.println("DD 的静态代码 1 被执行...");
}
{
System.out.println("DD 的普通代码块...");
}
}
class Animal {
static {
System.out.println("Animal 的静态代码 1 被执行...");
}
}
class Cat extends Animal {
public static int n1 = 999;
static {
System.out.println("Cat 的静态代码 1 被执行...");
}
}
class BB {
static {
System.out.println("BB 的静态代码 1 被执行...");
}
}
class AA extends BB {
static {
System.out.println("AA 的静态代码 1 被执行...");
}
}
4:创建一个对象时,在一个类中的调用顺序(难点)
- 调用静态代码块和静态属性初始化:静态代码块和静态属性的调用优先级一样,如果有多个代码块和多个静态变量初始化,则按他们的顺序调用
- 调用普通代码块和普通属性的初始化:两者调用的优先级一样,如果有多个普通代码块和多个普通属性初始化,则按他们定义的顺序来调用
class A {
{
System.out.println("A 普通代码块 01");
}
private int n2 = getN2();
static {
System.out.println("A 静态代码块 01");
}
private static int n1 = getN1();
public static int getN1() {
System.out.println("getN1 被调用...");
return 100;
}
public int getN2() {
System.out.println("getN2 被调用...");
return 200;
}
public A() {
System.out.println("A() 构造器被调用");
}
}
- 构造器的最前面其实隐含了super()和调用普通代码块,静态代码块和静态属性初始化,在类的加载时候,就执行完成,因此优于构造器和普通代码块执行的
class A {
public A() {
System.out.println("ok");
}
}
public class Code{
public static void main(String[] args) {
new BBB();
}
}
class AAA {
{
System.out.println("AAA 的普通代码块");
}
public AAA() {
System.out.println("AAA() 构造器被调用....");
}
}
class BBB extends AAA {
{
System.out.println("BBB 的普通代码块...");
}
public BBB() {
System.out.println("BBB() 构造器被调用....");
}
}
- 总结:创建一个子类父类对象时候(继承关系),他们的静态代码,静态属性初始化,普通代码块,普通属性初始化,构造方法的调用顺序如下:
- 1:父类的静态方法和的静态属性(优先级一样,按定义的顺序)
- 2:子类的静态代码块和静态属性(优先级一样,按定义的顺序)
- 3:父类的普通代码块和普通属性初始化(优先级一样,按定义的顺序执行)
- 4:父类的构造方法
- 5:子类的普通代码块和普通属性初始化(优先级一样,按定义的顺序执行)
- 6:子类的构造方法
class Sample
{
Sample(String s)
{
System.out.println(s);
}
Sample()
{
System.out.println("Sample 默认构造函数被调用");
}
}
public class Test{
Sample sam1=new Sample("sam1 成员初始化");
static Sample sam=new Sample("静态成员 sam 初始化 ");
static{
System.out.println("static 块执行");
if(sam==null)System.out.println("sam is null");
}
Test()
{
System.out.println("Test 默认构造函数被调用");
}
public static void main(String str[])
{
Test a=new Test();
}
}