单一职责原则(Single Responsibility Principle)

单一职责原则(SRP:The Single Responsibility Principle)

一个类应该有且只有一个变化的原因。

There should never be more than one reason for a class to change.

为什么将不同的职责分离到单独的类中是如此的重要呢?

因为每一个职责都是一个变化的中心。当需求变化时,这个变化将通过更改职责相关的类来体现。

如果一个类拥有多于一个的职责,则这些职责就耦合到在了一起,那么就会有多于一个原因来导致这个类的变化。对于某一职责的更改可能会损害类满足其他耦合职责的能力。这样职责的耦合会导致设计的脆弱,以至于当职责发生更改时产生无法预期的破坏。

例如,考虑下图中的设计。类图中显示 Rectangle 类包含两个方法,一个方法(Draw)负责在显示屏幕上绘制矩形,另一个方法(Area)负责计算矩形图形面积。

有两个不同的应用程序均使用了 Rectangle 类。一个应用为计算几何程序,它使用了 Rectangle 中的数学几何模型,但不会在显示屏幕上绘制矩形。另一个应用是一个图形界面程序(GUI),它可能会做一些计算几何方面的工作,但主要功能是在屏幕上绘制矩形。

1   public class Rectangle
 2   {
 3     public int Height { get; set; }
 4     public int Width { get; set; }
 6     public double Area()
 7     {
 8       return Width * Height;
 9     }
11     public void Draw(Form form)
12     {
13       SolidBrush brush = new SolidBrush(Color.Red);
14       Graphics formGraphics = form.CreateGraphics();
15       formGraphics.FillRectangle(brush,
16         new System.Drawing.Rectangle(
17           new Point(0, 0), new Size(Width, Height)));
18     }
19   }

这个设计侵犯了 SRP 原则。Rectangle 类包含了两个职责。第一个职责是提供矩形几何计算的数学模型,第二个职责是在 GUI 上渲染矩形。

对 SRP 原则的侵犯会导致诸多难以解决的问题:

首先,我们必须在计算几何应用中包含对 GUI 库的引用。这导致应用程序无谓的消耗了链接时间、编译时间、内存空间和存储空间等。

再者,如果因为某些原因对 GraphicalApplication 的一个更改导致 Rectangle 类也相应做了更改,这将强制我们对 ComputationalGeometryApplication 进行重新编译、重新测试和重新部署等。如果我们忘了做这些事情,那么应用程序可能以无法预期的方式而崩溃。

1   public class ComputationalGeometryApplication
 2   {
 3     public double CalculateArea(Rectangle rectangle)
 4     {
 5       double area = rectangle.Area();
 6       return area;
 7     }
 8   }
10   public class GraphicalApplication
11   {
12     public Form form { get; set; }
14     public void DrawOnScreen(Rectangle rectangle)
15     {
16       rectangle.Draw(form);
17     }
18   }

一个较好的设计是将这两个职责完全地隔离到不同的类当中,如下图所示。这个设计将 Rectangle 中关于几何计算的职责移到了 GeometricRectangle 类中,而 Rectangle 类中仅保留矩形渲染职责。

1   public class GeometricRectangle
 2   {
 3     public int Height { get; set; }
 4     public int Width { get; set; }
 6     public double Area()
 7     {
 8       return Width * Height;
 9     }
10   }
12   public class Rectangle
13   {
14     public void Draw(Form form, GeometricRectangle geometric)
15     {
16       SolidBrush brush = new SolidBrush(Color.Red);
17       Graphics formGraphics = form.CreateGraphics();
18       formGraphics.FillRectangle(brush,
19         new System.Drawing.Rectangle(
20           new Point(0, 0),
21           new Size(geometric.Width, geometric.Height)));
22     }
23   }

然后,如果我们再对 Rectangle 中渲染职责进行更改时将不会再影响到 ComputationalGeometryApplication 了。

1   public class ComputationalGeometryApplication
 2   {
 3     public double CalculateArea(GeometricRectangle geometric)
 4     {
 5       double area = geometric.Area();
 6       return area;
 7     }
 8   }
10   public class GraphicalApplication
11   {
12     public Form form { get; set; }
14     public void DrawOnScreen(Rectangle rectangleDraw, GeometricRectangle rectangleShape)
15     {
16       rectangleDraw.Draw(form, rectangleShape);
17     }
18   }

那么,职责(Responsibility)到底是什么?

单一职责原则(SRP:Single Responsibility Principle)的概念中,我们将职责(Responsibility)定义为 “一个变化的原因(a reason for change)”。如果你能想出多于一种动机来更改一个类,则这个类就包含多于一个职责。

职责的耦合有时很难被发现,因为我们习惯于将多个职责一起来考虑。例如,我们考虑下面定义的 Camera 接口,可能会认为这个接口看起来是非常合理的。接口中声明的 4 个方法从属于一个 Camera 接口定义。

1   public interface Camera
2   {
3     void Connect(string host);
4     void Disconnect();
5     void Send(byte[] data);
6     byte[] Receive();
7   }

然而,它确实耦合了 2 个职责。第一个职责是连接管理,第二个职责是数据通信。Connect 和 Disconnect 方法负责管理 Camera 与管理端 Host 的连接,而 Send 和 Receive 方法则负责收发通信数据。

这两个职责应该被分离吗?答案基本上是肯定的。这两组方法基本上没有任何交集,它们都可以依据不同的原因而变化。进一步说,它们将在应用程序中完全不同的位置被调用,而那些不同的位置将同样会因不同的原因而变化。

因此,下图中的设计可能会好一些。它将这两个职责分别隔离到不同的接口定义中,这至少使应用程序从两个职责中解耦。

然而,我们注意到这两个职责又重新被耦合进了一个 CameraImplementation 类中。这可能不是我们想要的,但却有可能是必须的。通常有很多原因会强制我们将一些职责耦合在一起。尽管如此,我们使得应用程序的其他部分得益于这个接口的隔离。

CameraImplementation 类在我们看来是一个组合出来的但确实包含一些缺点的类。但需要注意到的是,所有其他需要使用 CameraImplementation 类的地方已经可以被接口进行隔离,我们仅需依赖所定义的单一职责的接口。而 CameraImplementation 仅在被实例化的位置才会出现。我们将丑陋的代码限制在一定的范围内,而不会泄露或污染应用程序的其他部分。

总结

单一职责原则(SRP:Single Responsibility Principle)可表述为 “一个类应该有且只有一个变化的原因(There should never be more than one reason for a class to change.)”。单一职责原则是一个非常简单的原则,但通常也是最难做的正确的一个原则。职责的联合是在实践中经常碰到的事情,从这些各不相同的职责中发现并隔离职责就是软件设计的真谛所在。我们所讨论的其他设计原则最终也会回归到这个原则上来。

面向对象设计的原则

 SRP

 单一职责原则

 Single Responsibility Principle

 OCP

 开放封闭原则

 Open Closed Principle

 LSP

 里氏替换原则

 Liskov Substitution Principle

 ISP

 接口分离原则

 Interface Segregation Principle

 DIP

 依赖倒置原则

 Dependency Inversion Principle

 LKP

 最少知识原则

 Least Knowledge Principle

参考资料

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 单一职责原则Single Responsibility Principle,SRP)是面向对象设计中的一项基本原则,它指出一个类或者模块应该有且只有一个单一的功能或职责。 具体来说,单一职责原则要求一个类或者模块只负责一项功能,而不要有多个不同的功能。这样做的好处是可以使得代码更加清晰、易于维护、易于测试,同时也能够提高代码的复用性和可扩展性。 单一职责原则可以应用于各种不同的设计层面,包括类、模块、函数等,它要求每个设计单元都应该有一个清晰的职责,而不是混杂着多个职责。例如,一个类应该只负责一个单一的业务逻辑,而不应该包含过多的控制逻辑或数据访问逻辑。 总之,单一职责原则是面向对象设计中非常重要的一项原则,它能够帮助我们编写出更加清晰、可维护、可测试的代码。 ### 回答2: 单一职责原则(The Single Responsibility Principle)是面向对象设计原则之一。它要求一个类应该只有一个引起变化的原因。 具体而言,单一职责原则意味着一个类或模块应该只负责一个特定的功能或关注点。换句话说,一个类应该只有一个单一的职责。 这个原则的目标是让软件系统更加灵活、可维护和可扩展,因为如果一个类只负责一个职责,那么对于这个职责的变化不会影响到其他的职责。此外,单一职责原则使得代码更加简洁和易懂,因为每个类或模块的功能都是明确的,容易理解和修改。 具体实践单一职责原则需要遵循以下几个原则: 1. 高内聚(High Cohesion):类内部的各个成员方法应该紧密相关。一个类应该只包含与其职责相关的方法和属性,不应该包含多个无关的功能。 2. 低耦合(Low Coupling):类与类之间的依赖要尽量减少。一个类应该尽量减少对其他类的依赖,只与其必要的协作对象进行交互。 3. 分离关注点(Separation of Concerns):将不同的职责分离到不同的类中。每个类应该专注于完成自身的职责,不涉及其他职责。 通过遵循单一职责原则,可以提高代码的可读性、可维护性和可扩展性。同时,它也能够提高代码的复用性,因为一个类只负责一个职责,可以被其他模块或系统复用。 总之,单一职责原则是一种促进代码设计和组织的原则,它强调将不同的职责分离开来,提高代码的模块性和可维护性。这是面向对象设计中的一个重要概念,在软件开发中具有广泛的应用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值