249

我知道这应该是一个超级简单的问题,但我已经为这个概念苦苦挣扎了一段时间。

我的问题是,如何在 C# 中链接构造函数?

我正在上我的第一个 OOP 课程,所以我只是在学习。我不明白构造函数链接是如何工作的或如何实现它,甚至不明白为什么它比只做没有链接的构造函数更好。

我会很感激一些带有解释的例子。

那么如何将它们串起来呢?
我知道有两个:

public SomeClass this: {0}

public SomeClass
{
    someVariable = 0
} 

但是你怎么用三个,四个等等呢?

同样,我知道这是一个初学者问题,但我很难理解这一点,我不知道为什么。

4

9 回答 9

387

您使用标准语法(使用this类似方法)来选择类中的重载

class Foo 
{
    private int id;
    private string name;

    public Foo() : this(0, "") 
    {
    }

    public Foo(int id, string name) 
    {
        this.id = id;
        this.name = name;
    }

    public Foo(int id) : this(id, "") 
    {
    }

    public Foo(string name) : this(0, name) 
    {
    }
}

然后:

Foo a = new Foo(), b = new Foo(456,"def"), c = new Foo(123), d = new Foo("abc");

另请注意:

  • 您可以使用链接到基本类型的构造函数base(...)
  • 您可以在每个构造函数中添加额外的代码
  • 默认(如果您不指定任何内容)是base()

对于“为什么?”:

  • 代码减少(总是一件好事)
  • 必须调用非默认的基本构造函数,例如:

    SomeBaseType(int id) : base(id) {...}
    

请注意,您也可以以类似的方式使用对象初始化器(无需编写任何内容):

SomeType x = new SomeType(), y = new SomeType { Key = "abc" },
         z = new SomeType { DoB = DateTime.Today };
于 2009-11-29T08:06:27.697 回答
61

我只是想向任何搜索这个的人提出一个有效的观点。如果您打算使用 4.0 (VS2010) 之前的 .NET 版本,请注意您必须创建如上所示的构造函数链。

但是,如果您留在 4.0 中,我有个好消息。您现在可以拥有一个带有可选参数的构造函数!我将简化 Foo 类示例:

class Foo {
  private int id;
  private string name;

  public Foo(int id = 0, string name = "") {
    this.id = id;
    this.name = name;
  }
}

class Main() {
  // Foo Int:
  Foo myFooOne = new Foo(12);
  // Foo String:
  Foo myFooTwo = new Foo(name:"Timothy");
  // Foo Both:
  Foo myFooThree = new Foo(13, name:"Monkey");
}

当您实现构造函数时,您可以使用可选参数,因为已经设置了默认值。

我希望你喜欢这节课!我简直不敢相信开发人员自 2004/2005 年以来一直在抱怨构造链接并且无法使用默认的可选参数!现在它在开发世界中花了很长时间,开发人员害怕使用它,因为它不会向后兼容。

于 2011-01-13T14:36:39.313 回答
31

最好用一个例子来说明这一点。成像我们有一个类 Person

public Person(string name) : this(name, string.Empty)
{
}

public Person(string name, string address) : this(name, address, string.Empty)
{
}

public Person(string name, string address, string postcode)
{
    this.Name = name;
    this.Address = address;
    this.Postcode = postcode;
}

所以这里我们有一个构造函数,它设置了一些属性,并使用构造函数链来允许你创建一个只有一个名字的对象,或者只是一个名字和地址。如果您创建一个只有名称的实例,这将发送一个默认值 string.Empty 到名称和地址,然后将 Postcode 的默认值发送到最终构造函数。

这样做可以减少您编写的代码量。实际上只有一个构造函数中有代码,您不会重复自己,因此,例如,如果您将 Name 从属性更改为内部字段,则只需更改一个构造函数 - 如果您在所有三个构造函数中设置该属性那将是改变它的三个地方。

于 2009-11-29T08:09:30.340 回答
15

“构造函数链”的用途是什么?
您可以使用它从另一个构造函数调用一个构造函数。

如何实现“构造函数链”?
在定义构造函数后使用“:this (yourProperties)”关键字。例如:

Class MyBillClass
{
    private DateTime requestDate;
    private int requestCount;

    public MyBillClass()
    {
        /// ===== we naming "a" constructor ===== ///
        requestDate = DateTime.Now;
    }
    public MyBillClass(int inputCount) : this()
    {
        /// ===== we naming "b" constructor ===== ///
        /// ===== This method is "Chained Method" ===== ///
        this.requestCount= inputCount;
    }
}

为什么有用?
重要的原因是减少编码,防止重复代码。例如初始化属性的重复代码假设类中的某些属性必须用特定值初始化(在我们的示例中,requestDate)。并且类有 2 个或更多构造函数。如果没有“构造函数链”,您必须在类的所有构造函数中重复初始化代码。

它是如何工作的?(或者,“构造函数链”中的执行顺序是什么)?
在上面的例子中,方法“a”将首先执行,然后指令序列将返回到方法“b”。换句话说,上面的代码等于下面的代码:

Class MyBillClass
{
    private DateTime requestDate;
    private int requestCount;

    public MyBillClass()
    {
        /// ===== we naming "a" constructor ===== ///
        requestDate = DateTime.Now;
    }
    public MyBillClass(int inputCount) : this()
    {
        /// ===== we naming "b" constructor ===== ///
        // ===== This method is "Chained Method" ===== ///

        /// *** --- > Compiler execute "MyBillClass()" first, And then continue instruction sequence from here
        this.requestCount= inputCount;
    }
}
于 2016-12-13T06:34:05.773 回答
12

我有一个日记课,所以我不会一次又一次地写设置值

public Diary() {
    this.Like = defaultLike;
    this.Dislike = defaultDislike;
}

public Diary(string title, string diary): this()
{
    this.Title = title;
    this.DiaryText = diary;
}

public Diary(string title, string diary, string category): this(title, diary) {
    this.Category = category;
}

public Diary(int id, string title, string diary, string category)
    : this(title, diary, category)
{
    this.DiaryID = id;
}
于 2012-07-18T16:13:31.180 回答
6

你是在问这个吗?

  public class VariantDate {
    public int day;
    public int month;
    public int year;

    public VariantDate(int day) : this(day, 1) {}

    public VariantDate(int day, int month) : this(day, month,1900){}

    public VariantDate(int day, int month, int year){
    this.day=day;
    this.month=month;
    this.year=year;
    }

}
于 2009-11-29T08:07:39.050 回答
4

所有这些答案都很好,但我想在构造函数上添加一点更复杂的初始化说明。

class SomeClass {
    private int StringLength;
    SomeClass(string x) {
         // this is the logic that shall be executed for all constructors.
         // you dont want to duplicate it.
         StringLength = x.Length;
    }
    SomeClass(int a, int b): this(TransformToString(a, b)) {
    }
    private static string TransformToString(int a, int b) {
         var c = a + b;
         return $"{a} + {b} = {c}";
    }
}

尽管没有这个静态函数也可以解决这个例子,静态函数允许更复杂的逻辑,甚至可以从其他地方调用方法。

于 2020-11-24T14:22:00.097 回答
2

我希望下面的示例能够对构造函数链接有所了解。
例如,我的用例在这里,您希望用户将目录传递给构造函数,用户不知道要传递的目录并决定让您分配默认目录。您加强并分配一个您认为可行的默认目录。

顺便说一句,我在这个例子中使用了 LINQPad,以防你想知道 *.Dump() 是什么。
干杯

void Main()
{

    CtorChaining ctorNoparam = new CtorChaining();
    ctorNoparam.Dump();
    //Result --> BaseDir C:\Program Files (x86)\Default\ 

    CtorChaining ctorOneparam = new CtorChaining("c:\\customDir");
    ctorOneparam.Dump();    
    //Result --> BaseDir c:\customDir 
}

public class CtorChaining
{
    public string BaseDir;
    public static string DefaultDir = @"C:\Program Files (x86)\Default\";


    public CtorChaining(): this(null) {}

    public CtorChaining(string baseDir): this(baseDir, DefaultDir){}

    public CtorChaining(string baseDir, string defaultDir)
    {
        //if baseDir == null, this.BaseDir = @"C:\Program Files (x86)\Default\"
        this.BaseDir = baseDir ?? defaultDir;
    }
}
于 2013-04-11T00:08:04.050 回答
2

构造函数链中还有一个重要的点:顺序。为什么?假设您有一个在运行时由一个期望它是默认构造函数的框架构造的对象。如果您希望能够在传递值的同时仍然能够在需要时传递构造函数参数,那么这非常有用。

例如,我可以有一个支持变量,由我的默认构造函数设置为默认值,但可以被覆盖。

public class MyClass
{
  private IDependency _myDependency;
  MyClass(){ _myDependency = new DefaultDependency(); }
  MYClass(IMyDependency dependency) : this() {
    _myDependency = dependency; //now our dependency object replaces the defaultDependency
  }
}
于 2015-12-18T12:31:04.977 回答