c# 如何用组合替代继承

 更新时间:2021年02月19日 14:41:38   作者:丹枫无迹  
这篇文章主要介绍了c# 如何用组合替代继承,帮助大家更好的理解和学习使用c#,感兴趣的朋友可以了解下

如果问面向对象的三大特性是什么,多数人都能回答出来:封装、继承、多态。

继承 作为三大特性之一,近来却越来越不推荐使用,更有极端的语言,直接语法中就不支持继承,例如 Go。这又是为什么呢?

为什么不推荐使用继承?

假设我们要设计一个关于鸟的类。

我们将“鸟类”定义为一个抽象类 AbstractBird。所有更细分的鸟,比如麻雀、鸽子、乌鸦等,都继承这个抽象类。

大部分鸟都会飞,那我们可不可以在 AbstractBird 抽象类中,定义一个 Fly() 方法呢?

答案是否定的。尽管大部分鸟都会飞,但也有特例,比如鸵鸟就不会飞。鸵鸟继承具有 Fly() 方法的父类,那鸵鸟就具有“飞”这样的行为,这显然不符合我们对现实世界中事物的认识。

解决方案一

在鸵鸟这个子类中重写 Fly() 方法,让它抛出异常。

public class AbstractBird
{
  public virtual void Fly()
  {
    Console.WriteLine("I'm flying.");
  }
}

//鸵鸟
public class Ostrich : AbstractBird
{
  public override void Fly()
  {
    throw new NotImplementedException("I can't fly.");
  }
}

这种设计思路虽然可以解决问题,但不够优美。因为除了鸵鸟之外,不会飞的鸟还有很多,比如企鹅。对于这些不会飞的鸟来说,我们都需要重写 Fly() 方法,抛出异常。

这违背了迪米特法则(也叫最少知识原则),暴露不该暴露的接口给外部,增加了类使用过程中被误用的概率。

解决方案二

通过 AbstractBird 类派生出两个更加细分的抽象类:会飞的鸟类 AbstractFlyableBird 和不会飞的鸟类 AbstractUnFlyableBird,让麻雀、乌鸦这些会飞的鸟都继承 AbstractFlyableBird,让鸵鸟、企鹅这些不会飞的鸟,都继承 AbstractUnFlyableBird 类。

此时,继承关系变成了三层,还行得通。

如果要再添加一个游泳 Swim() 的方法,那情况就复杂了,要分为四中情况:

  • 会飞会游泳
  • 会飞不会游泳
  • 不会飞会游泳
  • 不会飞不会游泳

如果再有其他行为加入,抽象类的数量就会几何级数增长。

我们要搞清楚某个类具有哪些方法、属性,必须阅读父类的代码、父类的父类的代码……一直追溯到最顶层父类的代码。

使用组合

针对“会飞”这样一个行为特性,我们可以定义一个 Flyable 接口,只让会飞的鸟去实现这个接口。针对会游泳,定义一个 Swimable 接口,会叫定义一个 Tweetable 接口。

public interface Flyable
{
  void Fly();
}

public interface Swimable
{
  void Swim();
}

public interface Tweetable
{
  void Tweet();
}

//麻雀
public class Sparrow : Flyable, Tweetable
{
  public void Fly() => Console.WriteLine("I am flying.");

  public void Tweet() => Console.WriteLine("!@#$%^&*……");
}

//企鹅
public class Penguin : Swimable, Tweetable
{
  public void Swim() => Console.WriteLine("I am swimming.");

  public void Tweet() => Console.WriteLine("!@#$%^&*……");
}

麻雀和企鹅都会叫,Tweet 实现了两遍,这是坏味道。我们可以用组合来消除这个坏味道。

public interface Flyable
{
  void Fly();
}

public interface Swimable
{
  void Swim();
}

public interface Tweetable
{
  void Tweet();
}

public class FlyAbility : Flyable
{
  public void Fly() => Console.WriteLine("I am flying.");
}

public class SwimAbility : Swimable
{
  public void Swim() => Console.WriteLine("I am swimming.");
}

public class TweetAbility : Tweetable
{
  public void Tweet() => Console.WriteLine("!@#$%^&*……");
}

//麻雀
public class Sparrow : Flyable, Tweetable
{
  FlyAbility flyAbility = new FlyAbility();
  TweetAbility tweetAbility = new TweetAbility();

  public void Fly() => flyAbility.Fly();

  public void Tweet() => tweetAbility.Tweet();
}

//企鹅
public class Penguin : Swimable, Tweetable
{
  SwimAbility swimAbility = new SwimAbility();
  TweetAbility tweetAbility = new TweetAbility();

  public void Swim() => swimAbility.Swim();

  public void Tweet() => tweetAbility.Tweet();
}

虽然现在主流的思想都是多用组合少用继承,但是从上面的例子可以看出,继承改写成组合意味着要做更细粒度的类的拆分,要定义更多的类和接口。类和接口的增多也就或多或少地增加代码的复杂程度和维护成本。所以,在实际的项目开发中,我们还是要根据具体的情况,来具体选择该用继承还是组合。

以上就是c# 如何用组合替代继承的详细内容,更多关于c# 组合替代继承的资料请关注脚本之家其它相关文章!

相关文章

  • C#策略模式(Strategy Pattern)实例教程

    C#策略模式(Strategy Pattern)实例教程

    这篇文章主要介绍了C#策略模式(Strategy Pattern),以一个简单的实例讲述了C#策略模式的实现方法,包括策略模式的用途以及具体实现方法,需要的朋友可以参考下
    2014-09-09
  • C# WPF数据绑定模板化操作的完整步骤

    C# WPF数据绑定模板化操作的完整步骤

    WPF中的数据绑定提供了很强大的功能,与普通的WinForm程序相比,其绑定功能为我们提供了很多便利,下面这篇文章主要给大家介绍了关于C# WPF数据绑定模板化操作的完整步骤,需要的朋友可以参考下
    2022-01-01
  • C#实现学生管理系统

    C#实现学生管理系统

    这篇文章主要为大家详细介绍了C#实现学生管理系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-08-08
  • C#中的多线程多参数传递详解

    C#中的多线程多参数传递详解

    第一种解决方案的原理是:将线程执行的方法和参数都封装到一个类里面。通过实例化该类,方法就可以调用属性来实现间接的类型安全地传递多个参数
    2014-01-01
  • C#操作读取、写入XML文档的实用方法

    C#操作读取、写入XML文档的实用方法

    这篇文章主要介绍了C#操作读取、写入XML文档的实用方法,即即用.NET本身提供的Deserialize和Serialize进行反序列化和序列化XML文档,感兴趣的小伙伴们可以参考一下
    2016-04-04
  • DevExpress SplitContainerControl用法总结

    DevExpress SplitContainerControl用法总结

    这篇文章主要介绍了DevExpress SplitContainerControl用法,对初学者有一定的参考借鉴价值,需要的朋友可以参考下
    2014-08-08
  • C#实现把彩色图片灰度化代码分享

    C#实现把彩色图片灰度化代码分享

    这篇文章主要介绍了C#实现把彩色图片灰度化代码分享,用在一些特殊场合中,需要的朋友可以参考下
    2014-08-08
  • UnityShader3实现2D描边效果

    UnityShader3实现2D描边效果

    这篇文章主要为大家详细介绍了UnityShader3实现2D描边效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-02-02
  • C#实现记事本查找与替换功能

    C#实现记事本查找与替换功能

    这篇文章主要为大家详细介绍了C#实现记事本查找与替换功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-03-03
  • C#.NET获取拨号连接的宽带连接方法

    C#.NET获取拨号连接的宽带连接方法

    这篇文章主要介绍了C#.NET获取拨号连接的宽带连接方法,实例演示了一个C#封装的ADSL拨号连接类及其使用方法,需要的朋友可以参考下
    2015-06-06

最新评论