构造方法


创建实例的时候,我们经常需要同时初始化这个实例的字段,例如:

  1. Person ming = new Person();
  2. ming.setName("小明");
  3. ming.setAge(12);

初始化对象实例需要3行代码,而且,如果忘了调用setName()或者setAge(),这个实例内部的状态就是不正确的。

能否在创建对象实例时就把内部字段全部初始化为合适的值?

完全可以。

这时,我们就需要构造方法。

创建实例的时候,实际上是通过构造方法来初始化实例的。我们先来定义一个构造方法,能在创建Person实例的时候,一次性传入nameage,完成初始化:

构造方法 - 图1

由于构造方法是如此特殊,所以构造方法的名称就是类名。构造方法的参数没有限制,在方法内部,也可以编写任意语句。但是,和普通方法相比,构造方法没有返回值(也没有void),调用构造方法,必须用new操作符。

默认构造方法

是不是任何class都有构造方法?是的。

那前面我们并没有为Person类编写构造方法,为什么可以调用new Person()

原因是如果一个类没有定义构造方法,编译器会自动为我们生成一个默认构造方法,它没有参数,也没有执行语句,类似这样:

  1. class Person {
  2. public Person() {
  3. }
  4. }

要特别注意的是,如果我们自定义了一个构造方法,那么,编译器就不再自动创建默认构造方法:

构造方法 - 图2

如果既要能使用带参数的构造方法,又想保留不带参数的构造方法,那么只能把两个构造方法都定义出来:

构造方法 - 图3

没有在构造方法中初始化字段时,引用类型的字段默认是null,数值类型的字段用默认值,int类型默认值是0,布尔类型默认值是false

  1. class Person {
  2. private String name; // 默认初始化为null
  3. private int age; // 默认初始化为0
  4. public Person() {
  5. }
  6. }

也可以对字段直接进行初始化:

  1. class Person {
  2. private String name = "Unamed";
  3. private int age = 10;
  4. }

那么问题来了:既对字段进行初始化,又在构造方法中对字段进行初始化:

  1. class Person {
  2. private String name = "Unamed";
  3. private int age = 10;
  4. public Person(String name, int age) {
  5. this.name = name;
  6. this.age = age;
  7. }
  8. }

当我们创建对象的时候,new Person("Xiao Ming", 12)得到的对象实例,字段的初始值是啥?

在Java中,创建对象实例的时候,按照如下顺序进行初始化:

  1. 先初始化字段,例如,int age = 10;表示字段初始化为10double salary;表示字段默认初始化为0String name;表示引用类型字段默认初始化为null

  2. 执行构造方法的代码进行初始化。

因此,构造方法的代码由于后运行,所以,new Person("Xiao Ming", 12)的字段值最终由构造方法的代码确定。

多构造方法

可以定义多个构造方法,在通过new操作符调用的时候,编译器通过构造方法的参数数量、位置和类型自动区分:

  1. class Person {
  2. private String name;
  3. private int age;
  4. public Person(String name, int age) {
  5. this.name = name;
  6. this.age = age;
  7. }
  8. public Person(String name) {
  9. this.name = name;
  10. this.age = 12;
  11. }
  12. public Person() {
  13. }
  14. }

如果调用new Person("Xiao Ming", 20);,会自动匹配到构造方法public Person(String, int)

如果调用new Person("Xiao Ming");,会自动匹配到构造方法public Person(String)

如果调用new Person();,会自动匹配到构造方法public Person()

一个构造方法可以调用其他构造方法,这样做的目的是便于代码复用。调用其他构造方法的语法是this(…)

  1. class Person {
  2. private String name;
  3. private int age;
  4. public Person(String name, int age) {
  5. this.name = name;
  6. this.age = age;
  7. }
  8. public Person(String name) {
  9. this(name, 18); // 调用另一个构造方法Person(String, int)
  10. }
  11. public Person() {
  12. this("Unnamed"); // 调用另一个构造方法Person(String)
  13. }
  14. }

练习

请给Person类增加(String, int)的构造方法:

构造方法 - 图4

构造方法 - 图5下载练习:给Person类增加(String, int)的构造方法 (推荐使用IDE练习插件快速下载)

小结

实例在创建时通过new操作符会调用其对应的构造方法,构造方法用于初始化实例;

没有定义构造方法时,编译器会自动创建一个默认的无参数构造方法;

可以定义多个构造方法,编译器根据参数自动判断;

可以在一个构造方法内部调用另一个构造方法,便于代码复用。

读后有收获可以支付宝请作者喝咖啡:

构造方法 - 图6