隐式传递参数给基本构造函数

Sea*_*son 2 c# constructor

我有一堂课,看起来像:

public class WidgetDao : AbstractDao<Widget>, IWidgetDao
{
    public WidgetDao(ISession session)
        : base (session)
    {

    }
}
Run Code Online (Sandbox Code Playgroud)

如果我尝试在代码中删除此构造函数,我的代码会出现构建错误,如下所示:

public IWidgetDao GetWidgetDao()
{
    return _widgetDao ?? (_widgetDao = new WidgetDao(_session));
}
Run Code Online (Sandbox Code Playgroud)

我有点理解为什么我收到构建错误的原因...没有WidgetDao带有一个参数的显式构造函数。然而,WidgetDao继承AbstractDao<Widget>它确实有一个带有一个参数的构造函数。

我不明白为什么 C# 编译器会在WidgetDao其用途是将参数传递给基类时要求显式构造函数。

有人可以帮我解释一下吗?

Ber*_*sch 5

不幸的是,没有办法自动执行此操作。C# 为您的对象提供的唯一隐式构造函数是默认的无参数构造函数,并且仅当您的代码中没有其他选择时它才会这样做。我稍后将在此基础上进行构建,但只是为了提供基础信息......

隐式MyClass()构造函数:

public class MyClass 
{
}
Run Code Online (Sandbox Code Playgroud)

在上面的情况下,编译器假设您需要一个公共无参数构造函数,它除了允许您创建对象之外实际上没有做任何事情。即使您没有创建构造函数,您也可以new MyClass()整天调用。

现在,当您创建自己的构造函数时会发生什么?

public class MyClass
{
    public MyClass(string name)
    {
        Name = name;
    }

    public string Name { get; private set; }
}
Run Code Online (Sandbox Code Playgroud)

在这种情况下,隐式无参数构造函数仍然存在,但现在它是私有的。这意味着您不能再打电话new MyClass(),并且需要提供姓名。

现在让我们的类成为一个抽象基类:

public abstract class AbstractMyClass
{
    protected AbstractMyClass(string name)
    {
        Name = name;
    }

    public string Name { get; private set; }

    public abstract void DoSomething();
}
Run Code Online (Sandbox Code Playgroud)

无法实例化抽象类,因此最好的做法是创建构造函数protected。即使您制作了它们,public您也无法直接访问它们。重要的是,同样的警告也成立:默认的无参数构造函数现在是私有的。任何实现此基类的类都必须提供一个值,name否则该类不会完全实例化。

public class SeansClass
{
    public SeansClass() : base("Sean") { }

    public void DoSomething() { }
}
Run Code Online (Sandbox Code Playgroud)

在这种情况下,我们重新引入了标准的无参数构造函数,并让它name为我们的类提供。这是完全有效的代码。 唯一需要注意的是,调用中提供的值base()必须是编译时常量或根据传递到类的参数计算得出。base()方法总是首先被评估。

C# 无法假设您实际上希望基类的所有实现都具有相同的构造函数。每个类都可以添加构造函数或为基类提供自己的值。这只是 C# 的限制。