我们知道构造函数分为 不带参构造函数和带参构造函数,不带参构造函数是默认的构造函数,他们在使用过程中有什么样的区别?各自有什么样的作用?
概要回答:
构造函数与类同名。
除了名字外,构造函数的另一个特别之处是没有返回值类型,这与返回值类型为void的函数不同。
- 静态构造函数没有修饰符修饰(public,private)。
- 静态构造函数没有参数。
- 静态构造函数前面必须是static 关键字。
- 静态构造函数中不能实例化实例变量。
- 静态函数的调用时机,是在类被实例化或者静态成员被调用的时候进行调用,并且是由.net框架来调用静态构造函数来初始化静态成员变量。
- 一个类中只能有一个静态构造函数。
- 无参数的静态构造函数和无参数的构造函数是可以并存的。因为他们一个属于类级别,一个属于实例级别,并不冲突。
- 静态构造函数只会被执行一次。并且是在特点5中的调用时机中进行调用。
- 就像如果没有在类中写构造函数,那么框架会为我们生成一个构造函数,那么如果我们在类中定义了静态变量,但是又没有定义静态构造函数,那么框架也会帮助我们来生成一个静态构造函数来让框架自身来调用。
有时,在一个类中有几个构造函数,以容纳某些可选参数,这些构造函数都包含一些共同的代码。
例如,下面的情况:
- class Car {
- private string description;
- private uint nWheels;
- public Car(string description, uint nWheels)
- {
- this.description = description;
- this.nWheels = nWheels;
- }
- public Car(string description)
- {
- this.description = description;
- this.nWheels = 4;
- } // etc.
- }
这两个构造函数初始化了相同的字段,显然,最好把所有的代码放在一个地方。C#有一个特殊的语法,称为构造函数初始化器,可以实现此目的:
- class Car {
- private string description;
- private uint nWheels;
- public Car(string description, uint nWheels)
- {
- this.description = description;
- this.nWheels = nWheels;
- }
- public Car(string description) : this(description, 4)
- { }
- // etc
- }
这里,this关键字仅调用参数最匹配的那个构造函数。注意,构造函数初始化器在构造函数之前执行。现在假定运行下面的代码: Car myCar = new Car("Proton Persona"); 在本例中,在带一个参数的构造函数执行之前,先执行带2个参数的构造函数(但在本例中,因为带一个参数的构造函数没有代码,所以没有区别)。
构造函数是指在创建类的对象时必须执行的一系列操作,他也可以被重载,在创建类时可以根据实际参数情况合理调用一个构造函数。这里就有一个问题了:如果每一个构造函数都要执行一些特定的操作,那我们不得不在每一个构造函数里面粘贴执行这些特定操作的代码,这就造成了代码的重复。
解决办法是仅在缺省实例构造函数中放入这一段代码,让每个重载实例构造函数通过构造函数初始化器来调用它。
: this ([<形参列表>])
用法示例:
class Account
{
private decimal balance;
private decimal interestRate;
private string holderName;
public Account(decimal initialBalance, decimal initialInterestRate, string initialHolderName)
{
balance = initialBalance;
interestRate = initialBalance;
holderName = initialHolderName;
}
public Account(string initialHolderName)
: this(0, 0, initialHolderName)
{
Console.WriteLine("Only initial holder name supplied.");
}
public Account()
: this(0, 0, "Not assigned yet.")
{
Console.WriteLine("Note: no initial information supplied.");
}
}
上例中,Account类的构造函数被重载,无论调用哪一个重载实例构造函数,下面这一段代码总被执行:
public Account(decimal initialBalance, decimal initialInterestRate, string initialHolderName)
{
balance = initialBalance;
interestRate = initialBalance;
holderName = initialHolderName;
}
使用构造函数初始化器,其参数表达式有几点要注意:
构造函数初始化器必须位于它的实例构造函数形式参数作用域之内,如;
public Account(string initialHolderName)
: this(0, 0, initialHolderName)
是可行的,而下面
public Account(string initialHolderName)
: this(0, this.interestRate, initialHolderName)
无效,this引用了一个实例变量,但在这里,任何对实例变量的引用是无效的,而且在这里不能使用this关键字。
最后一点是,构造函数初始化器的参数列表不能与他所应用的实例构造函数的形式参数列表匹配,如下面的代码错误:
class test
{
...
public test(int initialValue) : this (10) //无效,initialValue与10匹配
{...}
...
}
错误的根本原因在于形成了一个无限自身调用链。