我正在使用.NET框架,但我确实希望能够制作一种供我所有网站使用的自定义类型的页面。当我尝试从控件访问页面时,问题就来了。我希望能够返回我的特定页面类型而不是默认页面。有什么办法吗?
public class MyPage : Page
{
// My own logic
}
public class MyControl : Control
{
public MyPage Page { get; set; }
}
Answers:
更新:这个答案写于2011年。经过二十多年的人们提出C#的返回类型协方差后,看起来终于可以实现了。我很惊讶。有关公告,请参见https://devblogs.microsoft.com/dotnet/welcome-to-c-9-0/的底部;我相信细节会跟着。
听起来您想要的是返回类型协方差。C#不支持返回类型协方差。
在返回类型协方差中,您可以覆盖一个基类方法,该方法返回一个特定性较低的类型,而一个返回特定性较高的类型:
abstract class Enclosure
{
public abstract Animal Contents();
}
class Aquarium : Enclosure
{
public override Fish Contents() { ... }
}
这是安全的,因为通过附件访问内容的消费者希望有动物,而水族馆承诺不仅要满足这一要求,而且要做出更严格的承诺:动物永远是鱼。
这种协方差在C#中不受支持,而且不太可能得到支持。CLR不支持它。(它受C ++和CLR上C ++ / CLI实现的支持;它是通过生成我在下面建议的那种神奇的辅助方法来实现的。)
(某些语言也支持形式参数类型的自变量-您可以使用采用Animal的方法重写采用Fish的方法。同样,合同已得到满足;基类要求处理任何Fish,派生该类承诺不仅可以处理鱼类,而且还可以处理任何动物。同样,C#和CLR不支持形式参数类型的自变量。
解决此限制的方法是执行以下操作:
abstract class Enclosure
{
protected abstract Animal GetContents();
public Animal Contents() { return this.GetContents(); }
}
class Aquarium : Enclosure
{
protected override Animal GetContents() { return this.Contents(); }
public new Fish Contents() { ... }
}
现在,您不仅获得了覆盖虚拟方法的好处,而且在使用编译时类型的Aquarium时获得了更强大的键入功能。
对于接口,我通过显式实现接口来解决:
public interface IFoo {
IBar Bar { get; }
}
public class Foo : IFoo {
Bar Bar { get; set; }
IBar IFoo.Bar => Bar;
}
是的,它支持协方差,但这取决于您要实现的确切目标。
我也倾向于将泛型用于很多事情,这意味着当您执行以下操作时:
class X<T> {
T doSomething() {
}
}
class Y : X<Y> {
Y doSomethingElse() {
}
}
var Y y = new Y();
y = y.doSomething().doSomethingElse();
而不是“丢掉”您的类型。
这是即将发布的C#9.0(.Net 5)的功能,您可以立即下载其预览版本。
下面的代码现在已经成功地构建(没有给出:error CS0508: 'Tiger.GetFood()': return type must be 'Food' to match overridden member 'Animal.GetFood()'
)
class Food { }
class Meat : Food { }
abstract class Animal {
public abstract Food GetFood();
}
class Tiger : Animal {
public override Meat GetFood() => default;
}
class Program {
static void Main() => new Tiger();
}
我还没有尝试过,但这行不通吗?
YourPageType myPage = (YourPageType)yourControl.Page;