C#继承和方法签名

时间:2013-06-20 16:11:24

标签: c# interface overloading

我正在研究一个需要根据我传入的对象类型运行不同进程方法的类。我认为重载可能在这里工作,但我有一个问题。假设我有两个接口:

public interface IEmail 
{
      Some properties ...
}

public interface ISpecialEmail : IEmail
{
     Some more properties....
}

和一个处理这些对象的类:

 public class EmailProcessor
 {

      public void ProcessEmail (IEmail email)
      {
           do stuff;
      }

      public void ProcessEmail (ISpecialEmail email)
      {

          do different stuff
      }
 }

我的问题是,由于ISpecialEmail继承自IEmail,这些方法签名是否足够不同以允许重载?我最初的想法是,ISpecialEmail电子邮件也会触发IEmail签名,因为从技术上讲,该接口也已实现。

感谢您的帮助。

2 个答案:

答案 0 :(得分:2)

这取决于你如何调用这些方法。

例如,假设您有Email : IEmailSpecialEmail : ISpecialEmail。如果您声明了电子邮件列表:

List<IEmail> emails = new List<IEmail> {new Email(), new SpecialEmail()};

然后跑

foreach (var email in emails) { EmailProcessor.ProcessEmail(email) }

它将为{em>两者调用public void ProcessEmail (IEmail email) - 因为调用绑定在编译时发生(即它不会按照您想要的方式工作)。

如果您执行以下操作,它也会失败:

var email = GetEmail(); // returns either IEmail or IExtendedEmail
EmailProcessor.ProcessEmail(email); // Would ONLY call ProcessEmail(IEmail)

因此,多态性会因这些签名而失败。

但是,以下方法可行:

var email = GetEmail(); // returns only IEmail
var extendedEmail = GetExtendedEmail(); // returns only IExtendedEmail
EmailProcessor.ProcessEmail(email); // Would all ProcessEmail(IEmail)
EmailProcessor.ProcessEmail(extendedEmail ); // Would call ProcessEmail(IExtendedEmail)

答案 1 :(得分:2)

根据C#规范(第7.4.3节)

“如果派生类中的任何方法适用,则基类中的方法不是候选者”

如果传入public void ProcessEmail (ISpecialEmail email)

ISpecialEmail应该是被调用的函数。但是,请注意将其转换为IEmail,因为这会改变优先级。如果您要将IEmailISpecialEmail两种类型的所有电子邮件合并到List IEmailIEmail进行处理,它们都将通过{{ 1}}函数的版本。