Вызов базового конструктора в C#
Если я наследую от базового класса и хочу передать что-то из конструктора унаследованного класса конструктору базового класса, как мне это сделать?
Например, если я наследую от класса Exception, я хочу сделать что-то вроде этого:
class MyExceptionClass : Exception
{
public MyExceptionClass(string message, string extraInfo)
{
//This is where it's all falling apart
base(message);
}
}
В основном я хочу иметь возможность передавать строковое сообщение в базовый класс Exception.
Ответов (12)12
Измените свой конструктор на следующее, чтобы он правильно вызывал конструктор базового класса:
public class MyExceptionClass : Exception
{
public MyExceptionClass(string message, string extrainfo) : base(message)
{
//other stuff here
}
}
Обратите внимание, что конструктор - это не то, что вы можете вызывать в любое время внутри метода. Вот почему вы получаете ошибки при вызове в теле конструктора.
Если вам нужно вызвать базовый конструктор, но не сразу, потому что ваш новый (производный) класс должен выполнять некоторые манипуляции с данными, лучшим решением будет прибегнуть к фабричному методу. Что вам нужно сделать, так это пометить свой производный конструктор как закрытый, затем создать статический метод в своем классе, который будет делать все необходимое, а затем вызывать конструктор и возвращать объект.
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);
}
}
Из Руководства по дизайну фреймворка и правил FxCop. :
1. Имя настраиваемого исключения должно оканчиваться на "Исключение".
class MyException : Exception
2. Исключение должно быть публичным.
public class MyException : Exception
3. CA1032: исключение должно реализовывать стандартные конструкторы.
- Открытый конструктор без параметров.
- Открытый конструктор с одним строковым аргументом.
- Открытый конструктор с одной строкой и исключением (поскольку он может обернуть другое исключение).
Конструктор сериализации защищен, если тип не запечатан, и закрытым, если тип запечатан. На основе MSDN :
[Serializable()] public 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. } protected MyException(SerializationInfo info, StreamingContext context) : base(info, context) { // Implement type-specific serialization constructor logic. } }
или
[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.
}
}
Верно использовать base
(что-то) для вызова конструктора базового класса, но в случае перегрузки используйте this
ключевое слово
public ClassName() : this(par1,par2)
{
// do not call the constructor it is called in the this.
// the base key- word is used to call a inherited constructor
}
// Hint used overload as often as needed do not write the same code 2 or more times
В соответствии с некоторыми другими ответами, перечисленными здесь, вы можете передавать параметры в конструктор базового класса. Рекомендуется вызывать конструктор базового класса в начале конструктора унаследованного класса.
public class MyException : Exception
{
public MyException(string message, string extraInfo) : base(message)
{
}
}
Я отмечаю, что в вашем примере вы никогда не использовали extraInfo
параметр, поэтому я предположил, что вы, возможно, захотите объединить extraInfo
строковый параметр со Message
свойством вашего исключения (кажется, что это игнорируется в принятом ответе и коде в вашем вопросе) .
Это просто достигается путем вызова конструктора базового класса и последующего обновления свойства Message дополнительной информацией.
public class MyException: Exception
{
public MyException(string message, string extraInfo) : base($"{message} Extra info: {extraInfo}")
{
}
}
Используя новые возможности C#, а именно out var
, вы можете избавиться от статического фабричного метода. Я только что обнаружил (случайно), что параметр var методов, называемых без base- "call", передается в тело конструктора.
Пример использования этого базового класса, от которого вы хотите унаследоваться:
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;
}
}
И решение с использованием статического частного вспомогательного метода, который генерирует все необходимые базовые аргументы (плюс дополнительные данные, если необходимо) и без использования статического фабричного метода, просто внешний конструктор:
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;
}
}
Обратите внимание, что вы можете использовать статические методы в вызове базового конструктора.
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;
}
}