Java基础之构造器、代码块、类加载时机的用法详解

 更新时间:2025年05月09日 09:51:46   作者:昊坤说不出的梦  
这篇文章主要介绍了Java基础之构造器、代码块、类加载时机的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

1、构造器细节

一个类可以定义多个不同的构造器,即构造器重载

  • 比如:我们可以再给Person类定义一个构造器,用来创建对象的时候,只指定人名不需要指定年龄。
  • 构造器名和类名要相同;
  • 构造器没有返回值;
  • 构造器是完成对象的初始化,并不是创建对象;
  • 创建对象时,系统自动的调用该类的构造方法;
  • 如果程序员没有定义构造器,系统会自动给类生成一个默认无参构造器(也叫默认构造器),比如 Dog (){}
  • 一旦定义了自己的构造器,默认的构造器就覆盖了,就不能再使用默认的无参构造器,除非显式的定义一下,即:Dog0){}

2、代码块细节

2.1 代码块

代码化块又称为初始化块,属于类中的成员[即:是类的一部分],类似于方法,将逻辑语句封装在方法体中,通过包围起来。

但和方法不同,没有方法名没有返回,没有参数,只有方法体,而且不用通过对象或类显式调用,而是加载类时,或创建对象时隐式调用

语法:

[修饰符]{
	//代码
	}

说明注意:

1)修饰符可选,要写的话,也只能写 static

2)代码块分为两类,使用static修饰的叫静态代码块,没有static修饰的,叫普通代码块/非静态代码块

3)逻辑语句可以为任何逻辑语句(输入、输出、方法调用、循环、判断等);

4)代码块尾部的;号可以写上,也可以省略。

理解:代码块相对于另一种形式的构造器(对构造器的补充机制),可以做初始化操作。

场景:如果多个构造器中有重复语句,可以抽取到代码块中,提高代码的复用性。

案例:

public 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;

    //3个构造器-》重载
    //解读
    //(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;
    }
}

2.2 静态代码块 & 类加载时机⭐⭐⭐

细节1

1)⭐static代码块也叫静态代码块,作用就是对类进行初始化,而且它随着类的加载而执行,并且只会执行一次。如果是普通代码块,每创建一个对象,就执行。

细节2

2)⭐类什么时候被加载

  • 创建对象实例时(new);
  • 创建子类对象实例,父类也会被加载;
  • 使用类的静态成员时(静态属性,静态方法);

案例1:

//1. 创建对象实例时(new)
AA aa = new AA();

class AA {
    //静态代码块
    static {
        System.out.println("AA 的静态代码1被执行...");
    }
}

AA 的静态代码1被执行...

案例2:

//2. 创建子类对象实例,父类也会被加载, 而且,父类先被加载,子类后被加载
AA aa2 = new AA();

class BB {
    //静态代码块
    static {
        System.out.println("BB 的静态代码1被执行...");//1
    }
}
class AA extends BB {
    //静态代码块
    static {
        System.out.println("AA 的静态代码1被执行...");//2
    }
}

BB 的静态代码1被执行...
AA 的静态代码1被执行...

案例3:

//3. 使用类的静态成员时(静态属性,静态方法)
System.out.println(Cat.n1);
 
class Animal {
    //静态代码块
    static {
        System.out.println("Animal 的静态代码1被执行...");//1
    }
}
class Cat extends Animal {
    public static int n1 = 999;//静态属性 3
    //静态代码块
    static {
        System.out.println("Cat 的静态代码1被执行...");//2
    }
}

Animal 的静态代码1被执行...
Cat 的静态代码1被执行...
999

细节3

3)普通的代码块,在创建对象实例时,会被隐式的调用,被创建一次,就会调用一次。

如果只是使用类的静态成员时,普通代码块并不会执行

小结:

  • 1、static代码块是类加载时,执行,只会执行一次;
  • 2、普通代码块是在创建对象时调用的,创建一次,调用一次。

细节4

4)创建一个对象时,在一个类调用顺序时:⭐⭐

  • 调用静态代码块和静态属性初始化(注意:静态代码块和静态属性初始化调用的优先级一样,如果有多个静态代码块和多个静态变量初始化,则按他们定义的顺序调用);
  • 调用普通代码块和普通属性的初始化(注意:普通代码块和普通属性初始化调用的优先级一样,如果有多个普通代码块和多个普通属性初始化,则按定义顺序调用);
  • 调用构造方法(即构造器)。

典型案例:

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() 构造器被调用

细节5

5)构造器的最前面其实隐含了super()和调用普通代码块,静态相关的代码块,属性初始化,在类加载时,就执行完毕,因此是优先于构造器和普通代码块执行的。

案例:

new 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() 构造器被调用....

细节6

6)创建含继承关系的一个子类对象时,他们的静态代码块,静态属性初始化,普通代码块,普通属性初始化,构造方法的调用顺序如下:⭐⭐⭐

  • 父类的静态代码块和静态属性(优先级一样,按定义顺序执行);
  • 子类的静态代码块和静态属性(优先级一样,按定义顺序执行);
  • 父类的普通代码块和普通属性初始化(优先级一样,按定义顺序执行);
  • 父类的构造方法(构造器);
  • 子类的普通代码块和普通属性初始化(优先级一样,按定义顺序执行;
  • 子类的构造方法(构造器)。

经典案例:

new B02();

class A02 { //父类
    private static int n1 = getVal01();
    static {
        System.out.println("A02的一个静态代码块..");//(2)
    }
    {
        System.out.println("A02的第一个普通代码块..");//(5)
    }
    public int n3 = getVal02();//普通属性的初始化
    public static int getVal01() {
        System.out.println("getVal01");//(1)
        return 10;
    }

    public int getVal02() {
        System.out.println("getVal02");//(6)
        return 10;
    }

    public A02() {//构造器
        //隐藏
        //super()
        //普通代码和普通属性的初始化......
        System.out.println("A02的构造器");//(7)
    }

}
class B02 extends A02 { //

    private static int n3 = getVal03();

    static {
        System.out.println("B02的一个静态代码块..");//(4)
    }
    public int n5 = getVal04();
    {
        System.out.println("B02的第一个普通代码块..");//(9)
    }

    public static int getVal03() {
        System.out.println("getVal03");//(3)
        return 10;
    }

    public int getVal04() {
        System.out.println("getVal04");//(8)
        return 10;
    }
    //一定要慢慢的去品..
    public B02() {//构造器
        //隐藏了
        //super()
        //普通代码块和普通属性的初始化...
        System.out.println("B02的构造器");//(10)
    }
}

细节7

7)静态代码块只能直接调用静态成员(静态属性和静态方法),普通代码块可以调用任意成员。

经典案例:

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();
    }
}

200
100
200

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 如何在JDK 9中更简洁使用 try-with-resources 语句

    如何在JDK 9中更简洁使用 try-with-resources 语句

    本文详细介绍了自 JDK 7 引入的 try-with-resources 语句的原理和用法,以及介绍了 JDK 9 对 try-with-resources 的改进,使得用户可以更加方便、简洁的使用 try-with-resources 语句。,需要的朋友可以参考下
    2019-06-06
  • Java编程实现游戏中的简单碰撞检测功能示例

    Java编程实现游戏中的简单碰撞检测功能示例

    这篇文章主要介绍了Java编程中的简单碰撞检测功能,涉及java针对坐标点的相关数学运算操作技巧,需要的朋友可以参考下
    2017-10-10
  • Java扑克牌速算24的方法

    Java扑克牌速算24的方法

    这篇文章主要为大家详细介绍了Java扑克牌速算24的方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03
  • 使用Java获取linux和window序列号

    使用Java获取linux和window序列号

    这篇文章主要为大家详细介绍了如何使用Java获取Windows和Linux系统上的CPU序列号、磁盘、mac地址等信息,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-01-01
  • java编程scanner类用法示例

    java编程scanner类用法示例

    这篇文章主要介绍了java编程scanner类用法示例,涉及一个通过scanner类实现需要手动输入变量时进行输入的实例,然后分享了一个简单的eclipse对Java代码格式化的技巧,具有一定借鉴价值,需要的朋友可以参考。
    2017-11-11
  • idea快速生成代码配置的方法示例

    idea快速生成代码配置的方法示例

    本文主要介绍了idea快速生成代码配置的方法示例,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-01-01
  • JVM---jstack分析Java线程CPU占用,线程死锁的解决

    JVM---jstack分析Java线程CPU占用,线程死锁的解决

    这篇文章主要介绍了JVM---jstack分析Java线程CPU占用,线程死锁的解决,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-09-09
  • SpringBoot 使用 Sa-Token 完成注解鉴权功能(权限校验)

    SpringBoot 使用 Sa-Token 完成注解鉴权功能(权限校验)

    Sa-Token 是一个轻量级 java 权限认证框架,主要解决登录认证、权限认证、单点登录、OAuth2、微服务网关鉴权 等一系列权限相关问题,这篇文章主要介绍了SpringBoot使用Sa-Token完成注解鉴权功能,需要的朋友可以参考下
    2023-05-05
  • Java自动取款机ATM案例实现

    Java自动取款机ATM案例实现

    本文主要介绍了Java自动取款机ATM案例实现,整个过程可以分为三部分:登录账户和执行取款操作,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-08-08
  • java实现带有背景图片的窗体

    java实现带有背景图片的窗体

    这篇文章主要为大家详细介绍了java实现带有背景图片的窗体,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-03-03

最新评论