C# 接口的默认实现(Default Interface Methods) - C# 8.0新特性

C# 8.0的接口默认实现允许在接口中提供方法体,使已有实现类无需修改即可使用新方法,解决接口演化时的编译兼容问题;新类可继承或重写默认方法,支持多接口组合,但不能访问私有成员、不支持状态保存,与抽象类有本质区别。

C# 8.0 引入的接口默认实现(Default Interface Methods)允许在接口中为方法提供具体实现,让实现类无需强制重写该方法即可直接使用。 这不是“抽象类替代方案”,而是为接口演化提供向后兼容的能力——比如给已有接口添加新方法时,老实现类不会因编译错误而崩溃。

为什么需要默认实现

过去给已发布的接口加新方法,所有实现类都必须立刻更新,否则编译失败。这在大型项目或公开 SDK 中极不现实。默认实现解决了这个问题:

  • 已有实现类可直接运行,不改代码也能用上新功能
  • 新实现类可选择继承默认行为,也可按需重写
  • 支持多继承式组合(一个类可实现多个带默认方法的接口)

语法和基本用法

在接口中用普通方法语法写实现体,不加 abstractvirtualsealed(这些关键字在接口默认方法中无效):

public interface ILogger
{
    void Log(s

tring message) => Console.WriteLine($"[Log] {message}");
string GetPrefix() => "APP";

// 静态方法、属性、字段、构造函数仍不支持

}

实现类可直接调用,也可用 base.InterfaceName.Method() 显式调用默认实现(C# 8+ 支持):

public class ConsoleLogger : ILogger
{
    public void Log(string message)
    {
        base.Logger.Log($"[CONSOLE] {message}"); // 显式调用接口默认实现
    }
}

注意事项和限制

默认方法看似灵活,但有明确边界:

  • 不能访问实现类的私有成员或字段(没有 this 上下文指向实例)
  • 不能声明 virtual / override —— 接口方法本身不参与虚方法表继承链
  • 若多个接口提供同签名默认方法,实现类必须显式实现该方法,否则编译报错(避免歧义)
  • 不支持默认实现的属性(get/set)、事件、运算符、索引器、构造函数

和抽象类的区别

默认接口方法 ≠ 抽象类:

  • 接口仍不能保存状态(无字段),抽象类可以
  • 类只能单继承抽象类,但可实现多个接口
  • 默认方法无法访问 this 的完整类型信息(比如不能调用子类特有方法)
  • 抽象类中的 virtual 方法支持运行时多态分发;接口默认方法是静态绑定到接口定义处的

基本上就这些。它不是为了鼓励“把逻辑全塞进接口”,而是为了解决真实世界中接口演进的兼容性难题——用得克制,效果最好。