🍀 基本介绍
代码化块又称为初始化块,属于类中的成员[即是类的一部分],类似于方法,将逻辑语句封装在方法体中,通过{} 包围起来。
但和方法不同,没有方法名,没有返回,没有参数,只有方法体,而且不用通过对象或类显式调用,而是加载类时,或创建对象时隐式调用。
🍀 基本语法
[修饰符]{
代码
};
??说明注意:
- 修饰符可选,要写的话,也只能写
static - 代码块分为两类,使用
static 修饰的叫静态代码块,没有static 修饰的,叫普通代码块/非静态代码块。 - 逻辑语句可以为任何逻辑语句(输入、输出、方法调用、循环、判断等)
; 号可以写上,也可以省略。
🍀代码块的好处和案例演示
- 相当于另外一种形式的构造器(对构造器的补充机制),可以做初始化的操作
- 场景:如果多个构造器中都有重复的语句,可以抽取到初始化块中,提高代码的重用性
??代码块的快速入门
public class CodeBlock01 {
public static void main(String[] args) {
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;
}
}
??运行结果:
电影屏幕打开...
广告开始...
电影正是开始...
Movie(String name)被调用
===============
电影屏幕打开...
广告开始...
电影正是开始...
Movie(String name, double price, String director)——)
🍀 代码块使用注意事项和细节讨论
??细节一: static代码块也叫静态代码块,作用就是对类进行初始化,而且它随着类的加载而执行,并且只会执行一次。如果是普通代码块,每创建一个对象,就执行。
??案例演示: 🚀static 代码块是在类加载时执行的,而且只会执行一次
public class CodeBlockDetail01 {
public static void main(String[] args) {
DD dd = new DD();
DD dd1 = new DD();
}
}
class DD {
static {
System.out.println("DD 的静态代码 1 被执行...");
}
}
DD 的静态代码 1 被执行...
??细节二: 类什么时候被加载[重要!要背!]
- 创建对象实例时(new)
- 创建子类对象实例,父类也会被加载
- 使用类的静态成员时(静态属性,静态方法)
??案例演示: 🚀1.创建对象实例时(new)
public class CodeBlockDetail01 {
public static void main(String[] args) {
AA aa = new AA();
}
}
class AA{
static{
System.out.println("AA的静态代码块被执行...");
}
}
AA的静态代码块被执行...
🚀2.创建子类对象实例,父类也会被加载,而且,父类先被加载,子类后被加载
public class CodeBlockDetail01 {
public static void main(String[] args) {
AA aa2 = new AA();
}
}
class BB {
static{
System.out.println("BB的静态代码块被执行...");
}
}
class AA extends BB{
static{
System.out.println("AA的静态代码块被执行...");
}
}
BB的静态代码块被执行...
AA的静态代码块被执行...
🚀3.使用类的静态成员时(静态属性,静态方法)
public class CodeBlockDetail01 {
public static void main(String[] args) {
System.out.println(Cat.n1);
}
}
class Cat{
public static int n1 = 999;
static {
System.out.println("Cat 的静态代码 1 被执行...");
}
}
Cat 的静态代码 1 被执行...
999
静态代码块的优先级大于静态变量
拓展:大家思考一个问题:如果Cat有一个父类Animal,当Cat的静态变量n1被调用时,Animal类有没有被加载?
public class CodeBlockDetail01 {
public static void main(String[] args) {
System.out.print(Cat.n1);
}
}
class Animal {
static {
System.out.println("Animal 的静态代码 1 被执行...");
}
}
class Cat extends Animal {
public static int n1 = 999;
static {
System.out.println("Cat 的静态代码 1 被执行...");
}
}
答案:会 可以看一下运行结果!
Animal 的静态代码 1 被执行...
Cat 的静态代码 1 被执行...
999
??细节三: 普通的代码块,在创建对象实例时,会被隐式的调用。被创建一次,就会调用一次。如果只是使用类的静态成员时,普通代码块并不会执行。
??案例演示: 🚀普通的代码块,在创建对象实例时,会被隐式的调用。被创建一次,就会调用一次。
public class CodeBlockDetail01 {
public static void main(String[] args) {
DD dd = new DD();
DD dd1 = new DD();
}
}
class DD {
static {
System.out.println("DD 的静态代码 1 被执行...");
}
{
System.out.println("DD 的普通代码块...");
}
}
DD 的静态代码 1 被执行...
DD 的普通代码块...
DD 的普通代码块...
🚀如果只是使用类的静态成员时,普通代码块并不会执行
public class CodeBlockDetail01 {
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 的普通代码块...");
}
}
DD 的静态代码 1 被执行...
8888
??小结: 1. static代码块是类加载时,执行,只会执行一次 2. 普通代码块是在创建对象时调用的,创建一次,调用一次 3. 类加载的3种情况,需要记住(细节二)
??细节四: 创建一个对象时,在一个类调用顺序是:(重点,难点);
- 调用静态代码块和静态属性初始化(注意:静态代码块和静态属性初始化调用的优先级一样,如果有多个静态代码块和多个静态变量初始化,则按他们定义的顺序调用)-见案例演示1
- 调用普通代码块和普通属性的初始化(注意:普通代码块和普通属性初始化调用的优先级一样,如果有多个普通代码块和多个普通属性初始化,则按定义顺序调用)见案例演示2
- 调用构造方法(构造器的优先级是最低的)-见案例演示3.
??案例演示: 🚀1. 调用静态代码块和静态属性初始化
代码一:
public class CodeBlockDetail02 {
public static void main(String[] args) {
A a = new A();
}
}
class A{
private static int n1 = getN1();
static {
System.out.println("A 静态代码块01");
}
public static int getN1(){
System.out.println("getN1被调用");
return 100;
}
}
??运行结果:
getN1被调用
A 静态代码块01
代码二(将静态方法和静态属性换个位置):
public class CodeBlockDetail02 {
public static void main(String[] args) {
A a = new A();
}
}
class A{
static {
System.out.println("A 静态代码块01");
}
private static int n1 = getN1();
public static int getN1(){
System.out.println("getN1被调用");
return 100;
}
}
??运行结果:
A 静态代码块01
getN1被调用
可以看到两段代码的运行结果是顺序不一样的
代码一:
getN1被调用
A 静态代码块01
代码二:
A 静态代码块01
getN1被调用
??说明: 静态代码块和静态属性初始化调用的优先级一样,如果有多个静态代码块和多个静态变量初始化,则按他们定义的顺序调用
??案例演示: 🚀2. 调用普通代码块和普通属性的初始化
代码一:
public class CodeBlockDetail02 {
public static void main(String[] args) {
A a = new A();
}
}
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;
}
}
??运行结果::
A 静态代码块 01
getN1 被调用...
A 普通代码块 01
getN2 被调用...
代码二(调换普通代码块和普通属性的顺序):
public class CodeBlockDetail02 {
public static void main(String[] args) {
A a = new A();
}
}
class A {
private int n2 = getN2();
{
System.out.println("A 普通代码块 01");
}
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;
}
}
??运行结果:
A 静态代码块 01
getN1 被调用...
getN2 被调用...
A 普通代码块 01
??说明: 普通代码块和普通属性初始化调用的优先级一样,如果有多个普通代码块和多个普通属性初始化,则按定义顺序调用
??案例演示: 调用构造方法
public class CodeBlockDetail02 {
public static void main(String[] args) {
A a = new A();
}
}
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() 构造器被调用");
}
}
??运行结果:
A 静态代码块 01
getN1 被调用...
A 普通代码块 01
getN2 被调用...
A() 构造器被调用
??注意??
静态相关的代码块,属性初始化,在类加载时,就执行完毕,因此是优先于构造器和普通代码块执行的
上面说的细节四一定要理解,要弄懂,要把顺序搞清楚,这是非常重要的!
??细节五: 构造器的最前面其实隐含了super()和调用普通代码块
class A {
public A( /构造器
System.out.println("ok");
}
}
??案例演示:
public class CodeBlockDetail03 {
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() 构造器被调用....");
}
}
??运行结果:
AAA 的普通代码块
AAA() 构造器被调用....
BBB 的普通代码块...
BBB() 构造器被调用....
??总结??
(父静>子静)>(父普>父构)>(子普>子构)
接下来是最为重量级的选手,大家打起精神来哦! ??细节六: 我们看一下创建一个子类对象时(继承关系),他们的静态代码块,静态属性初始化,普通代码块,普通属性初始化,构造方法的调用顺序如下:
①父类的静态代码块和静态属性初始化(优先级一样,按定义顺序执行) ②子类的静态代码块和静态属性初始化(优先级一样,按定义顺序执行) ③父类的普通代码块和普通属性初始化(优先级一样,按定义顺序执行) ④父类的构造方法 ⑤子类的普通代码块和普通属性初始化(优先级一样,按定义顺序执行 ⑥子类的构造方法
??注意:面试可能会问到这样的问题。 ??细节七:静态代码块只能直接调用静态成员(静态属性和静态方法),普通代码块可以调 用任意成员。
??案例演示:
class C02 {
private int n1 = 100;
private static int n2 = 200;
private void m1() {
}
private static void m2() {
}
static {
System.out.println(n2);
m2();
}
{
System.out.println(n1);
System.out.println(n2);
m1();
m2();
}
}
学习这些知识虽然比较麻烦,但是将来我们工作以后就会相对轻松了。
🍀练习题
??T1: 下面的代码输出什么?
class Person {
public static int total;
static {
total = 100;
System.out.println("in static block!");
}
}
public class Test {
public static void main(String[] args) {
System.out.println("total = "+ Person.total);
System.out.println("total = "+ Person.total);
}
}
??答案:
in static block!
total = 100
total = 100
??T2: 重点题:下面的代码输出什么?
public class Exercise{
public static void main(String str[])
{
Test a=new Test();
}
}
class Sample
{
Sample(String s)
{
System.out.println(s);
}
Sample()
{
System.out.println("Sample 默认构造函数被调用");
}
}
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 默认构造函数被调用");
}
}
??讲解: ① 在主方法中,创建一个Test对象时会先进行类的加载
public static void main(String str[])
{
Test a=new Test();
}
我们可以看到在Test类中首先进行的就是静态成员的初始化
static Sample sam=new Sample("静态成员 sam 初始化 ");
这时又会创建一个Sample类的对象,并且调用有参构造器 Sample(String s)
Sample(String s)
{
System.out.println(s);
}
所以首先输出的是:静态成员 sam 初始化 ②静态属性初始化完以后就是进行静态代码块的初始化
static{
System.out.println("static 块执行");
if(sam==null)System.out.println("sam is null");
}
这是会输出:static 块执行 ,由于sam已经不为空了,所以不会输出sam is null ;
③加载完有关静态的以后会加载构造器Test()
Test()
{
System.out.println("Test 默认构造函数被调用");
}
由于在构造器中隐含有super()和普通代码块或普通属性初始化的调用,所以会先调用普通属性初始化:
Sample sam1=new Sample("sam1 成员初始化");
这时就会输出语句:sam1 成员初始化 最后再输出语句:Test 默认构造函数被调用
??运行结果:
静态成员 sam 初始化
static 块执行
sam1 成员初始化
Test 默认构造函数被调用
|