⭐️代码块⭐️
🍀 基本介绍
代码化块又称为初始化块,属于类中的成员[即是类的一部分],类似于方法,将逻辑语句封装在方法体中,通过{}
包围起来。
但和方法不同,没有方法名,没有返回,没有参数,只有方法体,而且不用通过对象或类显式调用,而是加载类时,或创建对象时隐式调用。
🍀 基本语法
[修饰符]{
代码
};
⚡️说明注意:
- 修饰符可选,要写的话,也只能写
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;
//三个构造器==》重载
//解读
//(1) 下面的三个构造器都有相同的语句
//(2) 这样代码看起来比较冗余
//(3) 这时我们可以把相同的语句,放入到一个代码块中,即可
//(4) 这样当我们不管调用哪个构造器,创建对象,都会先调用代码块的内容
//(5) 代码块调用的顺序优先于构造器...
{
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) {
//类被加载的情况举例
//1. 创建对象实例时(new)
AA aa = new AA();
}
}
class AA{
//静态代码块
static{
System.out.println("AA的静态代码块被执行...");
}
}
AA的静态代码块被执行...
🚀2.创建子类对象实例,父类也会被加载,而且,父类先被加载,子类后被加载
public class CodeBlockDetail01 {
public static void main(String[] args) {
//类被加载的情况举例
//1. 创建对象实例时(new)
//AA aa = new AA();
//2. 创建子类对象实例,父类也会被加载, 而且,父类先被加载,子类后被加载
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) {
//类被加载的情况举例
//1. 创建对象实例时(new)
//AA aa = new AA();
//2. 创建子类对象实例,父类也会被加载, 而且,父类先被加载,子类后被加载
//AA aa2 = new AA();
//3. 使用类的静态成员时(静态属性,静态方法)
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 被执行...");//
}
//普通代码块, 在 new 对象时,被调用,而且是每创建一个对象,就调用一次
//可以这样简单的,理解 普通代码块是构造器的补充
{
System.out.println("DD 的普通代码块...");
}
}
DD 的静态代码 1 被执行...
DD 的普通代码块...
DD 的普通代码块...
🚀如果只是使用类的静态成员时,普通代码块并不会执行
public class CodeBlockDetail01 {
public static void main(String[] args) {
// 如果只是使用类的静态成员时,普通代码块并不会执行
System.out.println(DD.n1);//8888, 静态模块块一定会执行
}
}
class DD {
public static int n1 = 8888;//静态属性
//静态代码块
static {
System.out.println("DD 的静态代码 1 被执行...");//
}
//普通代码块, 在 new 对象时,被调用,而且是每创建一个对象,就调用一次
//和类加载没有关系,可以这样简单的,理解 普通代码块是构造器的补充
{
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( /构造器
//这里有隐藏的执行要求
//(1) super)://这个知识点,在前面讲解继承的时候总结过
//(2)调用普通代码块的
System.out.println("ok");
}
}
⚡️案例演示:
public class CodeBlockDetail03 {
public static void main(String[] args) {
new BBB();//(1)AAA 的普通代码块(2)AAA() 构造器被调用(3)BBB 的普通代码块(4)BBB() 构造器被调用
}
}
class AAA { //父类 Object
{
System.out.println("AAA 的普通代码块");
}
public AAA() {
//(1)super()
//(2)调用本类的普通代码块
System.out.println("AAA() 构造器被调用....");
}
}
class BBB extends AAA {
{
System.out.println("BBB 的普通代码块...");
}
public BBB() {
//(1)super()
//(2)调用本类的普通代码块
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(n1);错误
System.out.println(n2);//ok
//m1();//错误
m2();
}
{
//普通代码块,可以使用任意成员
System.out.println(n1);
System.out.println(n2);//ok
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 默认构造函数被调用
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/199744.html