来自:韩顺平老师的基础课程讲解
[修饰符] {
代码
};
注意:
代码块的好处和案例演示:
案例:CodeBlock01.java
package com.alan.codeblock; /** * 开发人员: Zzw * 开发时间: 2021/6/20 9:39 * 文件名称: CodeBlock01 * 开发工具: IntelliJ IDEA * 项目功能: 代码块入门 */ public class CodeBlock01 { public static void main(String[] args) { Movie movie1 = new Movie(); System.out.println("=========="); Movie movie2 = new Movie("潘叔劝你..."); System.out.println("=========="); Movie movie3 = new Movie("嘎子,你把握不住",66); } } class Movie{ private String name; private double price; // 代码块 { System.out.println("电影前戏"); System.out.println("电影开始"); } public Movie(){ System.out.println("Movie()..."); } public Movie(String name) { System.out.println("Movie(String name)..."); this.name = name; } public Movie(String name, double price) { System.out.println("Movie(String name, double price)..."); this.name = name; this.price = price; } }
输出:
电影前戏 电影开始 Movie()... ========== 电影前戏 电影开始 Movie(String name)... ========== 电影前戏 电影开始 Movie(String name, double price)...
=> 对于普通代码块,在创建(new)一个对象时,不管调用的哪个构造器,都会先执行一遍代码块的内容。
案例:CodeBlockDetail01.java
package com.alan.codeblock; import org.junit.Test; /** * 开发人员: Zzw * 开发时间: 2021/6/20 11:20 * 文件名称: CodeBlockDetail01 * 开发工具: IntelliJ IDEA * 项目功能: 代码块的使用细节 */ public class CodeBlockDetail01 { public static void main(String[] args) { // 类加载的情况举例 // 1.创建该类对象时 // 静态代码块只执行一次 在类加载时执行 Dog dog = new Dog(); // 2.创建子类对象实例 父类也会被加载 而且父类先加载 子类再加载 Son son = new Son(); // 3.使用类的静态成员(静态属性和静态方法) System.out.println("Dog.name: " + Dog.name); } } class Dog{ public static String name = "哈哈"; { System.out.println("Dog的 普通代码块 执行..."); } static { System.out.println("Dog的 静态代码块 执行..."); } } class Father { static { System.out.println("老子的 静态代码块 执行..."); } } class Son extends Father { public static String name = "儿子"; static { System.out.println("儿子的 静态代码块 执行..."); } }
输出:
Dog的 静态代码块 执行... Dog的 普通代码块 执行... Dog的 普通代码块 执行... 老子的 静态代码块 执行... 儿子的 静态代码块 执行... Dog.name: 哈哈
解释:
创建一个对象时,在一个类调用顺序是:(重点,难点)
案例:CodeBlockDetail02.java
package com.alan.codeblock; /** * 开发人员: Zzw * 开发时间: 2021/6/20 14:31 * 文件名称: CodeBlockDetail02 * 开发工具: IntelliJ IDEA * 项目功能: */ public class CodeBlockDetail02 { public static void main(String[] args) { Number number = new Number(); } } class Number { public Number() { System.out.println("Number() 构造器..."); } public int num1 = getNum1(); public int getNum1() { System.out.println("getNum1() 普通函数调用..."); return 233; } { System.out.println("普通代码块 调用..."); } public static int num2 = getNum2(); static { System.out.println("静态代码块 调用..."); } public static int getNum2() { System.out.println("getNum2() 静态函数调用..."); return 688; } }
输出:
getNum2() 静态函数调用... 静态代码块 调用... getNum1() 普通函数调用... 普通代码块 调用... Number() 构造器...
注:可以通过查看 Number.class文件 => 普通代码块是放在构造器中,并先执行,详情继续往下读。
Number.class
package com.alan.codeblock; class Number { public int num1 = this.getNum1(); public static int num2 = getNum2(); public Number() { System.out.println("普通代码块 调用..."); System.out.println("Number() 构造器..."); } public int getNum1() { System.out.println("getNum1() 普通函数调用..."); return 233; } public static int getNum2() { System.out.println("getNum2() 静态函数调用..."); return 688; } static { System.out.println("静态代码块 调用..."); } }
无参构造器的最前面其实隐含了super()和调用普通代码块,静态相关的代码块,属性初始化,在类加载时,就执行完毕,因此是优先于构造器和普通代码块执行的。
class A { public A() { // 隐藏的代码 // 1. super(); // 2.调用普通代码块 System.out.println("A 的无参构造器..."); } }
案例:CodeBlockDetail03.java
package com.alan.codeblock; /** * 开发人员: Zzw * 开发时间: 2021/6/20 15:10 * 文件名称: CodeBlockDetail03 * 开发工具: IntelliJ IDEA * 项目功能: 构造器与普通代码块 */ public class CodeBlockDetail03 { public static void main(String[] args) { new Student(); } } class Person { { System.out.println("Person 的普通代码块被执行..."); } public Person() { // super(); // 调用普通代码块 System.out.println("Person 的无参构造器被执行..."); } } class Student extends Person{ { System.out.println("Student 的普通代码块被执行..."); } public Student() { // super(); // 调用普通代码块 System.out.println("Student 的无参构造器被执行..."); } }
输出:
Person 的普通代码块被执行... Person 的无参构造器被执行... Student 的普通代码块被执行... Student 的无参构造器被执行...
解释:在创建(new) Student 类时,调用顺序:
静态代码块只能直接调用静态成员(静态属性和静态方法),普通代码块可以调用任意成员。
这个也很好理解哦: