我有一个要求,我有许多类都从单个基类派生而来。基类包含子类列表,这些子类也从相同的基类派生。
所有类都需要能够获取特定的值,这些值可以从类本身获得-或-取决于派生类的种类,它是其父类。
我看过使用方法而不是属性,但是我还想使这些值可用于.NET报告组件,该组件可直接访问报告引擎中公开的公共属性,因此不包括方法的使用。
我的问题是,在BaseClass中没有公开可用的setter的情况下在DerivedClass中实现setter的“最佳实践”方法是什么?
public class BaseClass
{
private BaseClass _Parent;
public virtual decimal Result
{
get { return ((_Parent != null) ? _Parent.Result : -1); }
}
}
public class DerivedClass : BaseClass
{
private decimal _Result;
public override decimal Result
{
get { return _Result; }
// I can't use a setter here because there is no set method in the base class so this throws a build error
//set { _Result = value; }
}
}
Run Code Online (Sandbox Code Playgroud)
我无法在BaseClass中添加受保护的setter(如下所示),因为无法更改DerivedClass中的访问修饰符。
public class BaseClass
{
private BaseClass _Parent;
public virtual decimal Result {
get { return ((_Parent != null) ? _Parent.Result : -1); }
protected set { }
}
}
public class DerivedClass : BaseClass
{
private decimal _Result;
public override decimal Result
{
get { return _Result; }
// This setter throws a build error because of a change of access modifiers.
//set { _Result = value; }
}
}
Run Code Online (Sandbox Code Playgroud)
我不想在带有Setter的BaseClass中添加成员变量,因为我不希望能够从BaseClass或其他从base派生的类中设置属性。
public class BaseClass
{
private BaseClass _Parent;
protected Decimal _Result; // This could result in a lot of unnecessary members in BaseClass.
public virtual decimal Result {
get { return _Result; }
// Empty setter ahead :) This does nothing.
// I could also throw an exception here but then issues would not be found until runtime
// and could cause confusion in the future with new developers etc.
set { }
}
}
public class DerivedClass : BaseClass
{
public override decimal Result
{
get { return base.Result; }
set { base._Result = value; }
}
}
Run Code Online (Sandbox Code Playgroud)
其他建议?
如果您可以在 BaseClass 中添加受保护的 setter,则可以在 DerivedClass 中使用 new 关键字:
public class BaseClass // ReadOnlyClass
{
public int Value { get; protected set; }
}
public class DerivedClass : BaseClass
{
public new int Value
{
get => base.Value;
set => base.Value = value;
}
}
Run Code Online (Sandbox Code Playgroud)
小智 4
您可以使用“new”关键字,但这将替换属性,您想要的内容是无法通过覆盖实现的。
public class BaseClass
{
private BaseClass _Parent;
public virtual decimal Result
{
get { return ((_Parent != null) ? _Parent.Result : -1); }
}
}
public class DerivedClass : BaseClass
{
private decimal _Result;
public new decimal Result
{
get { return _Result; }
set { _Result = value; }
}
}
Run Code Online (Sandbox Code Playgroud)