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
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持代码网。
发表评论