什么是构造器 构造器通常也叫构造方法、构造函数,构造器在每个项目中几乎无处不在。当你new一个对象时,就会调用构造器。构造器格式如下:
[修饰符,比如public] 类名 (参数列表,可以没有参数){ //这里不能有return }默认构造器 如果没有定义构造器,则会默认一个无参构造器,这就是为什么你定义了一个对象,比如 People,没有定义任何构造器却可以new这个对象,比如 new People() 。如果自定义了构造器,则会覆盖默认构造器。
如何禁止对象被外部创建 一些特殊需求,不希望定义的对象被外部创建(典型的就是单例了),那直接将构造器的修饰符改为 private 即可。这样就不能在外部通过new来创建这个对象了。
构造器重载 与普通方法一样,构造器也支持重载。一个对象中是可以支持同时定义多个构造器,通过不同的参数列表来实现重载。经常看到代码中new一个对象时,有时传入参数,有时又可以不用传。比如:new People()跟new People("张三"),这里就是重载了。
构造器的继承 子类构造器会默认调用父类无参构造器,如果父类没有无参构造器,则必须在子类构造器的第一行通过 super关键字指定调用父类的哪个构造器,具体看下文例子。final类是不允许被继承的,编译器会报错。很好理解,由于final修饰符指的是不允许被修改,而继承中,子类是可以修改父类的,这里就产生冲突了,所以final类是不允许被继承的。
构造器、静态代码块、构造代码块的执行顺序,详见下文实例
无继承的情况下的执行顺序静态代码块:只在程序启动后执行一次,优先级最高构造代码块:任何一个构造器被调用的时候,都会先执行构造代码块,优先级低于静态代码块构造器:优先级低于构造代码块总结一下优先级:静态代码块 > 构造代码块 > 构造器有继承的情况下的执行顺序:父类静态代码块:只在程序启动后执行一次,优先级最高子类静态代码块:只在程序启动后执行一次,优先级低于父类静态代码块父类构造代码块:父类任何一个构造器被调用的时候,都会执行一次,优先级低于子类静态代码块父类构造器:优先级低于父类构造代码子类构造代码块:子类任何一个构造器被调用的时候,都会执行一次,优先级低于父类构造器子类构造器:优先级低于子类构造代码块总结一下优先级:父类静态代码块 > 子类静态代码块 > 父类构造代码块 > 父类构造器 > 子类构造代码块 > 子类构造器新建一个类,不提供任何构造器,编译器会默认提供一个无参构造器,这就是为什么没定义任何构造器,却可以new 某个对象()
public class People { }以上这个People类,可以直接通过 new People()来实例化。
如果不希望People在外部通过new People()来实例化,只需要将构造器定义为private
public class People { private People(){ } }重载可以简单理解为:同个方法名,不同的参数列表。如果希望People能在外部通过new People() 或 new People("字符串") 来实例化,则通过以下代码即可
public class People { //通过new People()调用 public People(){ } //通过new People("字符串") 调用 public People(String str){ } }定义父类构造器,由于该构造器自定义了一个带参构造器,覆盖了默认的无参构造器,所以不能直接 new SuperClass() 调用了,除非再定义一个无参构造器
/** * 父类构造器 */ public class SuperClass { /** * 自定义带参构造器 */ public SuperClass(String str){ System.out.println("父类的带参构造方法,参数为:" + str); } }定义子类构造器,继承SuperClass,由于SuperClass没有无参构造器,所以必须在子类构造器中通过 super("字符串")来调用,否则编译器会报错
/** * 子类构造器 */ public class SubClass extends SuperClass { /** * 无参构造器 */ public SubClass(){ //由于SuperClass没有无参构造器,所以必须在子类构造器中通过 super("字符串")来调用,否则编译器会报错。 //如果没定义该句,则编译器会默认调用 super() super(""); } /** * 带参构造器 */ public SubClass(String subStr){ //由于SuperClass没有无参构造器,所以必须在子类构造器中通过 super("字符串")来调用,否则编译器会报错。 //如果没定义该句,则编译器会默认调用 super() super(subStr); } }实例化People
public static void main(String[] args){ System.out.println("--------------people----------------"); People people = new People(); System.out.println("--------------people1----------------"); People people1 = new People("张三"); }执行以上代码,输出:
--------------people---------------- 静态代码块,程序启动后执行,只会执行一次 构造代码块,每次调用构造方法都会执行一次 默认构造器 --------------people1---------------- 构造代码块,每次调用构造方法都会执行一次 带参构造器,参数:张三定义父类SuperClass
/** * 父类构造器 */ public class SuperClass { { System.out.println("父类构造代码块,每次调用构造方法都会执行的"); } /** * 父类无参构造方法 */ public SuperClass(){ System.out.println("父类的默认构造方法"); } /** * 重载,自定义父类带参构造方法 * @param str */ public SuperClass(String str){ System.out.println("父类的带参构造方法,参数为:" + str); } static { System.out.println("父类的静态代码块,程序启动后执行,只会执行一次"); } }定义子类SubClass,继承SuperClass
/** * 子类构造器,继承SuperClass */ public class SubClass extends SuperClass { static { System.out.println("子类的静态代码块,程序启动后执行,只会执行一次,先执行父类的,再执行子类的"); } { System.out.println("子类构造代码块,每次调用构造方法都会执行的"); } /** * 无参构造器 */ public SubClass(){ //这里没有指定调用父类哪个构造器,会默认调用 super(),调用父类的无参构造器public SuperClass() } /** * 重载构造器,多传两个参数 * @param str * @param str1 */ public SubClass(String str,String str1){ //必须写在构造器第一行,调用父类构造器 public SuperClass(String str) super(str); System.out.println("子类带参构造器:" + str1); } }实例化SubClass
public static void main(String[] args){ System.out.println("--------------subClass1----------------"); SubClass subClass1 = new SubClass(); System.out.println("--------------subClass2----------------"); SubClass subClass2 = new SubClass("子类第一个参数","子类第二个参数"); }执行以上代码,输出:
--------------subClass1---------------- 父类的静态代码块,程序启动后执行,只会执行一次 子类的静态代码块,程序启动后执行,只会执行一次,先执行父类的,再执行子类的 父类构造代码块,每次调用构造方法都会执行的 父类的默认构造方法 子类构造代码块,每次调用构造方法都会执行的 --------------subClass2---------------- 父类构造代码块,每次调用构造方法都会执行的 父类的带参构造方法,参数为:子类第一个参数 子类构造代码块,每次调用构造方法都会执行的 子类带参构造器:子类第二个参数