4

当我结合接口和多态性时,我一团糟。

假设我有以下界面:

public Interface CanSayHello
{
     String SayHello();
}

以下类:

public Class Person : CanSayHello
{
     public String SayHello() { return "Hey, I'm a person just saying hello to you";}
}

最后是重要的类:

public Class PoshPerson: Person
{
     public String SayHello() { return "Hey, I'm too posh to say hello to you";}
}

我的第一个问题是下面的代码会收集 PoshClass 或 Person Class 的方法吗?

public delegate String Collector();
event Collector CollectorEvent;
void GetMethod(CanSayHello c){CollectorEvent += c.SayHello;}

**GetMethod(new PoshPerson());**

如果它会从 Person 类中收集方法,我想我应该将 Person 的方法声明为 virtual 并重写 PoshPerson 的方法。我真的很希望这两个方法签名是相等的。有没有可能?

4

3 回答 3

4

您的示例运行Person's 方法。如果您更改PoshPersonPoshPerson : Person, CanSayHello(您可能认为不会改变任何内容,因为Person : CanSayHello),它会运行PoshPerson's 方法。

我同意@millimoose:“我会尽可能避免方法隐藏,正是因为编译时多态性不是很直观。” 我建议您更改Person.SayHellovirtual和。这样,无论您是否知道实例为、或,当前实例类型的方法都会运行。PoshPerson.SayHellooverrideCanSayHelloPersonPoshPerson

于 2012-06-24T12:31:11.007 回答
1

我认为你想要更多的东西:

public interface ITalkable
{ 
     string SayHello(); 
     string SayGoodbye();
     etc...
}

public class Person : ITalkable
{ 
     public virtual string SayHello() { return "Hey, I'm a person just saying hello to you";} 
     public virtual string SayGoodbye() { return "Hey, I'm a person just saying goodbye to you";}
} 

public class PoshPerson: Person 
{ 
     public override string SayHello() { return "Hey, I'm too posh to say hello to you";}
     public string MakePersonSayHello() { return base.SayHello(); } 

} 
于 2012-06-24T12:45:03.200 回答
1

您的示例将运行 Person 的版本。请参阅此处的说明:http: //msdn.microsoft.com/en-us/library/aa664593 (v=vs.71).aspx 。如果您在派生类中重新定义方法(编译后的 PoshPerson.SayHello 就像您使用 new 重新定义它一样),并且基类实现了一个接口,那么重新定义的方法不会改变接口映射。

如果你让 PoshPerson 重新实现接口,它会调用 PoshPerson 的 SayHello。

public class PoshPerson : Person, CanSayHello
{
    public String SayHello() { return "Hey, I'm too posh to say hello to you";}
}

或者,如果您在 Person 中将 SayHello 设为虚拟,然后在 PoshPerson 中覆盖它,它将调用 PoshPerson 的 SayHello。

于 2012-06-24T12:37:29.093 回答