如果我从基类继承,并想将继承类的构造函数传递给基类的构造函数,我该怎么做?

例如,如果我从Exception类继承,我希望执行以下操作:

class MyExceptionClass : Exception
{
     public MyExceptionClass(string message, string extraInfo)
     {
         //This is where it's all falling apart
         base(message);
     }
}

基本上,我希望能够将字符串消息传递给基础Exception类。


当前回答

public class MyExceptionClass : Exception
{
    public MyExceptionClass(string message,
      Exception innerException): base(message, innerException)
    {
        //other stuff here
    }
}

可以将内部异常传递给其中一个构造函数。

其他回答

您还可以使用构造函数中的参数进行条件检查,这允许一些灵活性。

public MyClass(object myObject=null): base(myObject ?? new myOtherObject())
{
}

or

public MyClass(object myObject=null): base(myObject==null ? new myOtherObject(): myObject)
{
}

注意,可以在对基构造函数的调用中使用静态方法。

class MyExceptionClass : Exception
{
     public MyExceptionClass(string message, string extraInfo) : 
         base(ModifyMessage(message, extraInfo))
     {
     }

     private static string ModifyMessage(string message, string extraInfo)
     {
         Trace.WriteLine("message was " + message);
         return message.ToLowerInvariant() + Environment.NewLine + extraInfo;
     }
}

例如,使用要从中派生的基类:

public abstract class BaseClass
{
    protected BaseClass(int a, int b, int c)
    {
    }
}

要执行的非编译伪代码:

public class DerivedClass : BaseClass
{
    private readonly object fatData;
    
    public DerivedClass(int m)
    {
        var fd = new { A = 1 * m, B = 2 * m, C = 3 * m };
        base(fd.A, fd.B, fd.C); // base-constructor call
        this.fatData = fd;
    }
}

2020版本(见下文,了解更严格的解决方案)

使用更新的C#特性,即out-var,您可以摆脱公共静态工厂方法。

我刚刚(偶然)发现,在base内部调用的方法的out var参数-“call”流到构造函数主体。(可能是C#的怪癖,请参阅2023版本的C#1.0兼容解决方案)

使用一个静态私有帮助器方法,它可以向外部生成所有必需的基本参数(如果需要,还可以添加其他数据),这只是一个公共的普通构造函数:

public class DerivedClass : BaseClass
{
    private readonly object fatData;
    
    public DerivedClass(int m)
        : base(PrepareBaseParameters(m, out var b, out var c, out var fatData), b, c)
    {
        this.fatData = fatData;
        Console.WriteLine(new { b, c, fatData }.ToString());
    }

    private static int PrepareBaseParameters(int m, out int b, out int c, out object fatData)
    {
        var fd = new { A = 1 * m, B = 2 * m, C = 3 * m };
        (b, c, fatData) = (fd.B, fd.C, fd); // Tuples not required but nice to use
        return fd.A;
    }
}

2023更新

您只需要一个额外的私有构造函数和一个附带的私有静态工厂方法,该方法使用与公共构造函数相同的输入为新的私有构造函数准备数据:

public class DerivedClass : BaseClass
{
    private readonly FatData fatData;

    public DerivedClass(int m)
        : this(PrepareBaseParameters(m))
    {
        Console.WriteLine(this.fatData.ToString());
    }

    private DerivedClass(FatData fd)
        : base(fd.A, fd.B, fd.C)
    {
        this.fatData = fd;
    }

    private static FatData PrepareBaseParameters(int m)
    {
        // might be any (non-async) code which e.x. calls factory methods 
        var fd = new FatData(A: 1 * m, B: 2 * m, C: 3 * m);
        return fd;
    }

    private readonly record struct FatData(int A, int B, int C);
}

不需要特殊的C#版本,C#10记录结构只是为了简短,也可以与任何C#1.0类一起使用。

这个版本似乎稍长,但它更容易阅读和理解。

来自框架设计指南和FxCop规则。:

1.自定义异常的名称应以异常结尾

    class MyException : Exception

2.异常应为公共

    public class MyException : Exception

3.CA1032:异常应实现标准构造函数。

公共无参数构造函数。具有一个字符串参数的公共构造函数。具有一个字符串和异常的公共构造函数(因为它可以包装另一个异常)。如果类型未密封,则受保护的序列化构造函数;如果类型已密封,则为私有构造函数。基于MSDN:[可串行化()]公共类MyException:异常{公共MyException(){//添加任何类型特定的逻辑,并提供默认消息。}公共MyException(字符串消息):基本(消息){//添加任何类型特定的逻辑。}公共MyException(字符串消息,异常innerException):基本(消息,内部异常){//为内部异常添加任何特定于类型的逻辑。}受保护的MyException(SerializationInfo信息,StreamingContext上下文):基本(信息,上下文){//实现特定于类型的序列化构造函数逻辑。}}

or

    [Serializable()]
    public sealed class MyException : Exception
    {
      public MyException()
      {
         // Add any type-specific logic, and supply the default message.
      }

      public MyException(string message): base(message) 
      {
         // Add any type-specific logic.
      }
      public MyException(string message, Exception innerException): 
         base (message, innerException)
      {
         // Add any type-specific logic for inner exceptions.
      }
      private MyException(SerializationInfo info, 
         StreamingContext context) : base(info, context)
      {
         // Implement type-specific serialization constructor logic.
      }
    }  

如果您需要调用基构造函数,但不是立即调用,因为您的新(派生)类需要进行一些数据操作,那么最好的解决方案是使用工厂方法。您需要做的是将派生构造函数标记为私有,然后在类中创建一个静态方法,该方法将完成所有必要的工作,然后调用构造函数并返回对象。

public class MyClass : BaseClass
{
    private MyClass(string someString) : base(someString)
    {
        //your code goes in here
    }

    public static MyClass FactoryMethod(string someString)
    {
        //whatever you want to do with your string before passing it in
        return new MyClass(someString);
    }
}