Java学习笔记——静态代码块、构造代码块、构造方法、普通代码块

参考:
https://www.cnblogs.com/ysocean/p/8194428.html
https://blog.csdn.net/qq_35868412/article/details/89360250

静态代码块

1.格式

在Java类中(方法中不能存在静态代码块)使用static关键字和{}声明的代码块:

public class CodeBlock {
    static{
        System.out.println("静态代码块");
    }
}
2.执行时机

静态代码块在类被加载的时候就运行了,而且只运行一次,并且优先于各种代码块以及构造函数。如果一个类中有多个静态代码块,会按照书写顺序依次执行。后面在比较的时候会通过具体实例来证明。

3.静态代码块的作用

一般情况下,如果有些代码需要在项目启动的时候就执行,这时候就需要静态代码块,比如一个项目启动需要加载的很多配置文件等资源,就可以都放入静态代码块中;静态代码块也可以初始化类变量,即static修饰的数据成员。

4.静态代码块不能存在任何方法体中

这个应该很好理解,首先我们要明确静态代码块是在类加载的时候就要运行了。我们分情况讨论:

对于普通方法,由于普通方法是通过加载类,然后new出实例化对象,通过对象才能运行这个方法,而静态代码块只需要加载类之后就能运行了。

对于静态方法,在类加载的时候,静态方法也已经加载了,但是我们必须要通过类名或者对象名才能访问,也就是说相比于静态代码块,静态代码块是主动运行的,而静态方法是被动运行的。

不管是哪种方法,我们需要明确静态代码块的存在在类加载的时候就自动运行了,而放在不管是普通方法还是静态方法中,都是不能自动运行的。

5.静态代码块不能访问普通变量

这个理解思维同上,普通变量只能通过对象来调用,是不能放在静态代码块中的。

构造代码块

1.格式

在Java类中使用{}声明的代码块(和静态代码块的区别是少了static关键字):

public class CodeBlock {
    static{
        System.out.println("静态代码块");
    }
    {
        System.out.println("构造代码块");
    }
}

它可以初始化类的实例变量。

2.执行时机

构造代码块在创建对象时被调用,每次创建对象都会调用一次,但是优先于构造方法执行。如果存在多个构造代码块,则执行顺序按照书写顺序依次执行。

需要注意的是,听名字我们就知道,构造代码块是依托于构造方法执行的,也即构造代码块会在构造方法执行时,在构造方法主体代码执行之前被运行,也就是说,如果你不实例化对象,构造代码块是不会执行的。怎么理解呢?我们看看下面这段代码:

public class CodeBlock {
    {
        System.out.println("构造代码块");
    }
     
    public CodeBlock(){
        System.out.println("无参构造函数");
    }
    public CodeBlock(String str){
        System.out.println("有参构造函数");
    }
}

反编译生成的class文件:
在这里插入图片描述

3.构造代码块的作用

和构造方法的作用类似,都能对对象进行初始化,并且只要创建一个对象,构造代码块都会执行一次。但是反过来,构造方法则不一定每个对象建立时都执行(多个构造方法情况下,建立对象时传入的参数不同则初始化使用对应的构造方法)。

因此,利用每次创建对象的时候都会提前调用一次构造代码块特性,可以初始化类的实例变量,也可以做诸如统计创建对象的次数等功能。

构造方法

1.构造方法的命名必须和类名完全相同。在Java中普通方法可以和构造方法同名,但是必须带有返回值;

public class Puppy{
    public Puppy(){
    }
 
    public Puppy(String name){
        // 这个构造器仅有一个参数:name
    }
}

2.构造方法的功能主要用于在类的对象创建时定义初始化的状态。它没有返回值,也不能用void来修饰。这就保证了它不仅什么也不用自动返回,而且根本不能有任何选择。而其他方法都有返回值,即使是void返回值。尽管方法体本身不会自动返回什么,但仍然可以让它返回一些东西,而这些东西可能是不安全的;

3.构造方法不能被直接调用,必须通过new运算符在创建对象时才会自动调用(创建对象时至少要调用一个构造方法);而一般的方法是在程序执行到它的时候被调用的;

4.每个类都有至少一个构造方法。当定义一个类的时候,通常情况下都会显式地为该类定义构造方法,并在方法中指定初始化的工作(也可省略);如果没有显示定义,Java 编译器将会为该类提供一个默认构造方法,此默认构造方法是不带参数的。而一般的方法不存在这一特点;

普通代码块

普通代码块和构造代码块的区别是,构造代码块是在中定义的,而普通代码块是在方法体中定义的。普通代码块的执行顺序和书写顺序一致。

public void sayHello(){
    {
        System.out.println("普通代码块");
    }
}

执行顺序

静态代码块>构造代码块>构造方法>普通代码块

例子:

public class CodeBlock {
    static{
        System.out.println("静态代码块");
    }
    {
        System.out.println("构造代码块");
    }
    public CodeBlock(){
        System.out.println("无参构造方法");
    }
    public void sayHello(){
        {
            System.out.println("普通代码块");
        }
    }

    public static void main(String[] args) {
        System.out.println("执行main方法");

        new CodeBlock().sayHello();
        System.out.println("---------------");
        new CodeBlock().sayHello();
    }
}

反编译生成的class文件:

public class CodeBlock {
    public CodeBlock() {
        System.out.println("构造代码块");
        System.out.println("无参构造方法");
    }

    public void sayHello() {
        System.out.println("普通代码块");
    }

    public static void main(String[] args) {
        System.out.println("执行main方法");
        (new CodeBlock()).sayHello();
        System.out.println("---------------");
        (new CodeBlock()).sayHello();
    }

    static {
        System.out.println("静态代码块");
    }
}

输出:

静态代码块
执行main方法
构造代码块
无参构造方法
普通代码块
---------------
构造代码块
无参构造方法
普通代码块

上述例子创建了两个匿名对象,但是静态代码块只调用了一次。

父类和子类执行顺序

父类的静态代码块>子类的静态代码块>父类的构造代码块>父类的构造方法>子类的构造代码块>子类的构造方法

总之一句话,静态代码块内容先执行,接着执行父类构造代码块和构造方法,然后执行子类构造代码块和构造方法。

注意,子类是不继承父类的构造方法的,它只是隐式或显式地调用。如果父类的构造方法带有参数,则必须在子类的构造方法中显式地通过 super 关键字调用父类的构造方法并配以适当的参数列表;如果父类构造方法没有参数,则在子类的构造方法中不需要使用 super 关键字调用父类构造方法,系统会自动调用父类的无参构造方法。

例子:

父类SuperClass:

public class SuperClass {
    static {
        System.out.println("父类静态代码块");
    }
    {
        System.out.println("父类构造代码块");
    }
    public SuperClass() {
        System.out.println("父类构造方法");
    }
}

子类SubClass,并在子类SubClass中执行main方法:

public class SubClass extends SuperClass{
    static{
        System.out.println("子类静态代码块");
    }
    {
        System.out.println("子类构造代码块");
    }
    public SubClass(){
        System.out.println("子类构造方法");
    }

    public static void main(String[] args){
        System.out.println("子类中的main方法");
        new SubClass();
        System.out.println("------------");
        new SubClass();
    }
}

输出:

父类静态代码块
子类静态代码块
子类中的main方法
父类构造代码块
父类构造方法
子类构造代码块
子类构造方法
------------
父类构造代码块
父类构造方法
子类构造代码块
子类构造方法

类似地,静态代码块只调用了一次。

例子2:

父类SuperClass,并在父类SuperClass中创建一个子类SubClass的对象:

public class SuperClass {
    static {
        System.out.println("父类静态代码块");
    }
    {
        System.out.println("父类构造代码块");
    }
    public SuperClass() {
        System.out.println("父类构造方法");
    }

    public static void main(String[] args){
        System.out.println("父类中的main方法");
        new SubClass();
    }
}

子类SubClass:

public class SubClass extends SuperClass{
    static{
        System.out.println("子类静态代码块");
    }
    {
        System.out.println("子类构造代码块");
    }
    public SubClass(){
        System.out.println("子类构造方法");
    }
}

输出:

父类静态代码块
父类中的main方法
子类静态代码块
父类构造代码块
父类构造方法
子类构造代码块
子类构造方法

静态代码块需要在加载类时才执行,因为子类对象的创建在执行父类中的main之后,因此父类中的main方法优于子类静态代码块。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值