freeCodeCamp/guide/chinese/java/constructors/index.md

181 lines
5.7 KiB
Markdown
Raw Normal View History

---
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)
// ##构造函数链接