开发者

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

目录
  • 1、构造器细节
  • 2、代码块细节
    • 2.1 代码块
    • 2.2 静态代码块 & 类加载时机⭐⭐⭐
      • 细节1
      • 细节2
      • 细节3
      • 细节4
      • 细节5
      • 细节6
      • 细节7
  • 总结

    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.ouHnYlpqt.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 stHnYlpqatic 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

    总结

    以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程客栈(www.devze.com)。

    0

    上一篇:

    下一篇:

    精彩评论

    暂无评论...
    验证码 换一张
    取 消

    最新开发

    开发排行榜