181 lines
5.7 KiB
Markdown
181 lines
5.7 KiB
Markdown
---
|
||
title: Constructors
|
||
localeTitle: 构造函数
|
||
---
|
||
如果一个对象从一个类复制,那有什么意义呢?我应该能够将数据存储在其中吗?
|
||
|
||
那时我们使用**getter** (例如,getName())/ **setter** (例如,setName())方法,或者在本例中使用构造函数来初始化类。基本上,每个Java类都有一个构造函数,它是在初始化类的任何对象时首先调用的方法。可以把它想象成一些入门代码。
|
||
|
||
当您编写没有任何构造函数的类时,Java编译器会创建一个默认构造函数:
|
||
|
||
```java
|
||
public class Car {
|
||
private String name;
|
||
}
|
||
|
||
Car modelS = new Car();
|
||
```
|
||
|
||
这种没有参数的初始化是一种调用默认构造函数的方法。您也可以使用这种方式编写默认构造函数:
|
||
|
||
```java
|
||
public class Car {
|
||
private String name;
|
||
|
||
// User Specified Default Constructor
|
||
public Car() {
|
||
name = "Tesla";
|
||
}
|
||
}
|
||
```
|
||
|
||
然后,当调用`new Car()` ,变量`name`自动初始化为Carla对象的该实例的“Tesla”。
|
||
|
||
显然,构造函数与它们完全一样:它们用于`construct` ie,实例化特定类的对象。
|
||
构造函数看起来类似于方法声明,但在它们的意义上略有不同:
|
||
|
||
1. 被命名与该类完全相同。
|
||
2. 没有返回类型。
|
||
|
||
因此,使用`constructors`的目的是提供:
|
||
|
||
1. 一种实例化对象的方法。
|
||
2. 为对象属性提供初始值。
|
||
3. 控制对象的创建方式。
|
||
|
||
让我们看另一个例子。比如说,本田(汽车制造商)希望所有汽车都被命名为`Honda <a name>` 。为了强制执行此操作,我们可以使用如下类来表示:
|
||
|
||
```java
|
||
public class Car {
|
||
|
||
private String name;
|
||
|
||
// Constructor.
|
||
public Car(String model){
|
||
this.name = "Honda " + model;
|
||
}
|
||
|
||
public String getName(){
|
||
return this.name;
|
||
}
|
||
|
||
public static void main(String args[]){
|
||
Car car = new Car("Civic");
|
||
System.out.println( car.getName() );
|
||
}
|
||
}
|
||
```
|
||
|
||
![:rocket:](//forum.freecodecamp.com/images/emoji/emoji_one/rocket.png?v=2 ":火箭:") [运行代码](https://repl.it/CTJ4/1)
|
||
|
||
请注意,当我们以这种方式编写构造函数(即提供参数)时,我们控制(第3点) `Car`实例的创建方式。简而言之,我们在这个例子中说, **你必须提供一个模型名称才能获得Car类的实例** 。
|
||
|
||
为什么这很重要?有时您希望在整个应用程序中使用`one and only one`类的实例。实现此目的的一种方法是使用`private`构造函数。
|
||
|
||
假设您需要一个班级来代表银行。您不希望人们创建`Bank`实例。所以,你设计你的课程:
|
||
|
||
```java
|
||
public class Bank {
|
||
|
||
private static Bank instance;
|
||
|
||
private Bank(){
|
||
}
|
||
|
||
public static Bank getInstance(){
|
||
if(null == instance){
|
||
instance = new Bank();
|
||
}
|
||
return instance;
|
||
}
|
||
}
|
||
```
|
||
|
||
![:rocket:](//forum.freecodecamp.com/images/emoji/emoji_one/rocket.png?v=2 ":火箭:") [运行代码](https://repl.it/CTJz/0)
|
||
|
||
请注意,构造函数是`private` 。这强制了这样一个事实,即不允许其他人创建银行的实例。
|
||
事实上,如果在另一个班级,你尝试:
|
||
|
||
```java
|
||
Bank account = new Bank(); // Throws a compilation error: Bank() has private access in Bank.
|
||
```
|
||
|
||
因此,获取实例访问权限的唯一方法是使用`Bank.getInstance()` 。这样的实例称为`Singleton`因为在应用程序的整个生命周期中,您只能获得一个实例(每个VM都是精确的)。
|
||
|
||
一个类中可以有许多构造函数。但它们应该在方法参数上有所不同。这是构造函数重载。确切地说,我们说当有两个或多个具有相同名称但方法参数不同的构造函数时,会发生构造函数重载。因此,这两个函数具有不同的方法签名,并且被Java视为完全不同的构造函数。例如:
|
||
|
||
```java
|
||
public class Car {
|
||
|
||
private String name;
|
||
private String carType;
|
||
|
||
// Constructor.
|
||
public Car(){
|
||
this.name = "No Name";
|
||
this.carType = "No Type";
|
||
}
|
||
public Car(String model){
|
||
this.name = "Honda " + model;
|
||
}
|
||
|
||
public Car(String model, String carType){
|
||
this.name = model;
|
||
this.carType = carType;
|
||
}
|
||
|
||
public String getName(){
|
||
return this.name;
|
||
}
|
||
|
||
public String getCarType(){
|
||
return this.name;
|
||
}
|
||
|
||
public static void main(String args[]){
|
||
Car car = new Car("Civic");
|
||
System.out.println( car.getName() );
|
||
|
||
// Other Way To Initialize
|
||
Car car = new Car("Civic","Sedan");
|
||
System.out.println( car.getName() + " "+ car.getCarType() );
|
||
|
||
}
|
||
}
|
||
```
|
||
|
||
因此,获取实例访问权限的唯一方法是使用`Bank.getInstance()` 。这样的实例称为`Singleton`因为在应用程序的整个生命周期中,您只能获得一个实例(每个VM都是精确的)。
|
||
|
||
## 复制构造函数
|
||
|
||
复制构造函数是一个构造函数,它通过使用先前创建的同一类的对象初始化它来创建对象。复制构造函数用于 -
|
||
|
||
1. 从另一个相同类型的对象初始化。
|
||
2. 复制对象以将其作为参数传递给函数。
|
||
3. 复制对象以从函数返回它。 这是一个程序,它显示了复制构造函数的简单用法:
|
||
|
||
```Java
|
||
class Complex {
|
||
|
||
private double re, im;
|
||
|
||
// A normal parametrized constructor
|
||
public Complex(double re, double im) {
|
||
this.re = re;
|
||
this.im = im;
|
||
}
|
||
|
||
// Copy constructor
|
||
Complex(Complex c) {
|
||
System.out.println("Copy constructor called");
|
||
re = c.re;
|
||
im = c.im;
|
||
}
|
||
|
||
}
|
||
}
|
||
```
|
||
|
||
[运行完整的代码](https://repl.it/MwnJ)
|
||
|
||
// ##构造函数链接 |