第一章:构造函数揭秘 ------ 创造者的第一次触碰
构造函数,顾名思义,是用于创建和初始化对象的特殊方法。它没有返回类型,名字与类名一致。构造函数是对象诞生的第一步,也是最至关重要的一步。让我们通过一个生动的例子来理解构造函数的魔力:
java
public class Car {
private String make;
private String model;
private int year;
// 构造函数
public Car(String make, String model, int year) {
this.make = make;
this.model = model;
this.year = year;
}
// Getter 和 Setter 方法省略...
}
在上述示例中,Car
类定义了汽车的基本属性:制造商(make)、型号(model)和生产年份(year)。构造函数接收这三个参数,并将它们赋值给类的相应属性。这就是创造者的第一次触碰,赋予了对象生命和特性。
第二章:构造函数重载 ------ 多元化的生命起源
构造函数重载提供了创建对象的多种途径,就像自然界中生命的多样性。在同一个类中,你可以定义多个构造函数,每个构造函数具有不同的参数列表。下面,我们将扩展 Car
类,加入更多的构造选项:
java
public class Car {
private String make;
private String model;
private int year;
private double price;
// 无参构造函数
public Car() {
this("Unknown", "Unknown", 1970, 0.0); // 使用默认值
}
// 构造函数重载
public Car(String make, String model) {
this(make, model, 1970, 0.0); // 使用默认的年份和价格
}
public Car(String make, String model, int year) {
this(make, model, year, 0.0); // 使用默认的价格
}
public Car(String make, String model, int year, double price) {
this.make = make;
this.model = model;
this.year = year;
this.price = price;
}
// Getter 和 Setter 方法省略...
}
现在,我们可以通过四种不同的构造函数来创建 Car
对象,每一种都代表了不同的生命起源。这种灵活性使得我们能够根据具体需求,轻松地初始化对象的不同状态。
第三章:构造函数链 ------ 继承的血脉传承
在Java中,子类不仅可以继承父类的属性和方法,还可以调用父类的构造函数。这称为构造函数链,它确保了子类对象的正确初始化。让我们通过一个继承的例子来说明这一点:
java
public class Vehicle {
protected String make;
protected String model;
// 构造函数
public Vehicle(String make, String model) {
this.make = make;
this.model = model;
}
}
public class Car extends Vehicle {
private int year;
private double price;
// 构造函数
public Car(String make, String model, int year, double price) {
super(make, model); // 调用父类构造函数
this.year = year;
this.price = price;
}
// Getter 和 Setter 方法省略...
}
在上面的代码中,Vehicle
类是所有交通工具的基础,而 Car
类继承了 Vehicle
类的属性和构造函数。Car
的构造函数首先调用了 super
关键字,这意味着它调用了父类 Vehicle
的构造函数,从而确保了所有继承自父类的属性都被正确初始化。
第四章:构造函数与封装 ------ 秘密花园的守门人
构造函数不仅是对象的生命之源,也是封装原则的守护者。通过构造函数,我们可以控制对象的初始化过程,限制外部直接访问和修改对象的内部状态。这就像花园的守门人,只有经过精心挑选的访客才能进入秘密花园。
在我们的 Car
类中,构造函数是唯一可以直接设置 make
、model
、year
和 price
属性的地方。通过将这些属性声明为私有(private),我们确保了外部代码不能直接修改这些属性,只能通过构造函数和公共的 getter/setter 方法间接访问。