深入理解程序设计|策略与机制分离


在现代操作系统的结构设计中,经常利用“机制与策略分离”的原理来构造OS结构。所谓机制,是指实现某一功能的具体执行机构。而策略,则是在机制基础上,借助于某些参数和算法来实现该功能的优化,或达到不同的功能目标。通常,机制处于一个系统的基层,而策略则处于系统的高层。

在程序设计中,机制与策略分离的思想可以提高程序的可复用性,可维护性和可调试性使程序更具有高内聚低耦合性。如果说机制是砖,那么策略就是房子,同样的砖可以建不同的房子,我们不能把建砖和建房子混在一起实现。

策略的变化要远远大于机制的变化。将两者分离,可以使机制相对保持稳定,而同时支持策略的变化。

在代码大全中提到“隔离变化”的概念,以及设计模式中提到的将易变化的部分和不易变化的部分分离也是这个思路。


欢迎大家加入极客星球圈子:

  • 修炼基本功:分享多年基础技术深度理解(比如深入理解系列),基础概念深度解析,经典书籍推荐和阅读经验分享,经典开源软件源码理解分析等;

  • 扩展技术和商业视野:真正的分享海内外技术发展,大厂技术内幕,业界解决方案;

  • 校招/社招:关于一切找工作问题分享,面试问题,简历修改,面试经验,面试问题答疑,各类方向后端学习路线和就业指导;

  • 职场普升/技术专家:分享各种不同公司宝贵的职场普升经验,核心员工,职场潜规则, 团队合作,做事经验,普升经验,技术深耕经验,少走弯路;

  • 专属交流群:高手交流,开阔眼界,分享心得,每天成长,厚积薄发,任何技术问题,任何疑难杂症,都可以咨询,讨论,交流!

深入理解系列

  • 深入理解计算机系统

  • 深入理解操作系统(调度,内存,网络,IO)

  • 深入理解并发技术全景指南

  • 深入理解编程语言

  • 深入理解算法与数据结构

  • 深入理解网络协议

  • 深入理解网络编程

  • 深入理解性能优化

  • 深入理解分布式技术

  • 深入理解数据库

  • 深入理解代码设计

  • 深入理解架构设计

详细了解:极客星球  ,现在加入超级优惠,早点加入一起学习成长,早点突破成长瓶颈:

433cde0f2019ebd9c11f0ed6871e0844.png

扫码加入,加我微信:fr35331508,拉你进VIP群。


在《Unix编程艺术》第一章就深刻讨论这个编程哲学:

“在我们对 Unix 错误的讨论中,我们观察到 X window的设计者做出了一个基本决定来实现“机制,而不是策略” —— 使 X 成为一个通用的图形引擎,并将有关用户界面风格的决定留给工具包和其他级别的系统。我们通过指出政策和机制倾向于在不同的时间尺度上发生变异来证明这一点,政策的变化比机制快得多,GUI 工具包的外观和感觉上的时尚可能来来去去,但光栅操作和合成是永恒的。

因此,将策略和机制硬连接在一起会产生两个负面影响:它使策略变得僵化并且更难以响应用户需求而改变,这意味着试图改变策略有很强的破坏机制稳定的倾向。

另一方面,通过将两者分开,我们可以在不破坏机制的情况下试验新策略。我们还使为机制编写好的测试变得更加容易。

实现这种分离的一种方法是,例如,将应用程序编写为由嵌入式脚本语言驱动的 C服务例程库,应用程序控制流是用脚本语言而不是 C 编写的。这种模式是Emacs编辑器,它使用嵌入式 Lisp解释器来控制用 C 编写的编辑原语。

另一种方法是将您的应用程序分成协作的前端和后端进程,这些进程通过套接字上的专用应用程序协议进行通信;前端执行策略,后端实现机制。这样的全局复杂性通常远低于实现相同功能的单进程单体的复杂性,从而减少您对错误的脆弱性并降低生命周期成本(提高健壮性)。”

一些例子

GUI框架

MVC(Model-View-Controller)作为最经典的GUI架构,MVC模式的核心思想是数据层(Domain)与表现层(Presentation)的隔离。

9cee5ebcdcb5303a9d0fcae737831e12.jpeg

  • 模型(Model) 用于封装与应用程序的业务逻辑相关的数据以及对数据的处理方法。“ Model ”有对数据直接访问的权力,例如对数据库的访问。“Model”不依赖“View”和“Controller”,也就是说, Model 不关心它会被如何显示或是如何被操作。但是 Model 中数据的变化一般会通过一种刷新机制被公布。为了实现这种机制,那些用于监视此 Model 的 View 必须事先在此 Model 上注册,从而,View 可以了解在数据 Model 上发生的改变。

  • 视图(View)能够实现数据有目的的显示(理论上,这不是必需的)。在 View 中一般没有程序上的逻辑。为了实现 View 上的刷新功能,View 需要访问它监视的数据模型(Model),因此应该事先在被它监视的数据那里注册。

  • 控制器(Controller)起到不同层面间的组织作用,用于控制应用程序的流程。它处理事件并作出响应。“事件”包括用户的行为和数据 Model 上的改变。

View,Model属于策略,在系统中属于可变部分,Controller属于机制,不会随着view的变化而变化,属于系统中不变的部分,构建一个系统要尽肯能分离可变部分和不可变部分。

netfilter框架

dc3cc39d3662a1fbde119f6466a3df1a.png

netfilter框架是一个典型将机制和策略分离好例子:

294399bd4ef1f98dd822cb30a4caa7ce.png

Netfilter是一个设计良好的框架,之所以说它是一个框架是因为它提供了最基本的底层支撑,而对于实现的关注度却没有那么高,这种底层支撑实际上是5个HOOK点:

PREROUTING:数据包进入网络层路由前

FORWARD:数据包路由之后确定要转发之后

INPUT:数据包路由之后确定要本地接收之后

OUTPUT:本地数据包发送

POSTROUTING:数据包发出去之前

238a7779035a5e706f8ddf2be37dc0f7.png

Netfilter拥有几乎无限的可扩展性, Liuux中使用的仅仅是它的一个很小的部分,大部分的内容作为可插拔的module处于待命状态Netfilter的机制集成在Linux内核中, 然而它的策略扩展却处于一个独立的空间,我们说这种所谓的机制也仅仅是5个HOOK点。

我们浏览netfilter.org就会知道,它里面融合了大量的策略,我们最熟悉的就是iptables了,上图的ebtables,arptables,nft也是Netfilter的扩展之一, 足以看出,Netfilter有多强大,内核仅仅给出钩子点而已, 如果你嫌某些不好,你可以自己实现一个更好的,事实上,Netfilter中有很多的东西并没有集成在Linux内核。

TCP拥塞控制框架

Linux系统中的TCP拥塞控制采用面向对象的设计思想,提供拥塞控制接口用于实现不同的拥塞控制策略,成功把拥塞控制解耦了,大大提高了内核TCP实现的可扩展性:

5deca8aaa1cfcace5cb1e9b6e7f0aa9d.png

eBPF框架

5f79e90618023573f93485603a4a21de.png

  • 内核实现BPF虚拟机执行核心引擎,属于机制部分;

fd5b69962def3f64628ec301a02e757a.png

  • 用户态可以编写各种BPF程序,实现不同策略功能;

游戏引擎

e2d18dc29beadf5e9c05cf829ce977b6.jpeg

游戏引擎便是专门为游戏而设计的工具及技术集成,之所以称为引擎,如同交通工具中的引擎,提供了最核心的技术部分--游戏机制,然后可以通过脚本语言或者关卡设计来插入策略逻辑.

46130ee7e59b20b398117ca5d14cebd3.png

游戏引擎架构

游戏引擎通常由运行时组件和工具套件两部分构成,  重用性是游戏引擎的一个重要设计目标,这样很多游戏开发都可以通过"换皮策略"来快速开发新游戏。

最后一些问题

1、透过现象看本质,机制与策略到底是什么?为什么要将机制与策略分离?

机制可以认为是业务通用的核心模型(框架),不易变化;策略可以认为是某个功能的具体实现方案,可以被框架使用;机制与策略分离,是一种可扩展性设计的重要方法,提供一个继承接口,用于提供不同的实现,这也就是策略模式和接口隔离原则。机制关联一个抽象的策略(也就是接口),用不同的具体策略初始化抽象策略,就能调用具体策略的处理流程。

2、假如不分离,会出现什么问题?

把策略同机制揉成一团有两个负面影响:一来会使策略变得死板,难以适应用户需求的改变,二来也意味着任何策略的改变都极有可能动摇机制,对原来稳定的框架造成污染,引入风险。

所以我们在设计系统的时候,可以参考这种机制和策略模式,让系统具有更好的扩展性和更好的稳定性。

参考和扩展阅读

https://web.archive.org/web/20050306210911/http://www.faqs.org/docs/artu/ch01s06.html#id2877777

https://qcc107.github.io/2015/09/01/UNIX%E7%BC%96%E7%A8%8B%E8%89%BA%E6%9C%AF%E4%B9%8B%E7%AD%96%E7%95%A5%E4%B8%8E%E6%9C%BA%E5%88%B6%E7%9B%B8%E5%88%86%E7%A6%BB/#:~:text=%E6%89%80%E8%B0%93%E6%9C%BA%E5%88%B6%EF%BC%8C%E6%98%AF%E6%8C%87%E5%AE%9E%E7%8E%B0,%E5%86%85%E8%81%9A%E4%BD%8E%E8%80%A6%E5%90%88%E6%80%A7%E3%80%82

602eb65eeff5cc794d3cafe14cbf62d6.jpeg

深入理解零拷贝技术


9ba4a1f96eea6ee89fb3ad4635ed60bb.jpeg

TCP协议疑难杂症全景解析|硬核


981bfc2056dfe77a2128d94f22e09c30.jpeg

腾讯全球招聘技术大咖


1f32db5239c5a1213e1987d61c648390.jpeg

2023 计算机最值得加入的100+家国企!


9e1308fc1172d9145881c5aef6f71020.jpeg

做完这些lab,国内外大厂横着走

欢迎大家加入极客星球圈子:

  • 修炼基本功:分享多年基础技术深度理解(比如深入理解系列),基础概念深度解析,经典书籍推荐和阅读经验分享,经典开源软件源码理解分析等;

  • 扩展技术和商业视野:真正的分享海内外技术发展,大厂技术内幕,业界解决方案;

  • 校招/社招:关于一切找工作问题分享,面试问题,简历修改,面试经验,面试问题答疑,各类方向后端学习路线和就业指导;

  • 职场普升/技术专家:分享各种不同公司宝贵的职场普升经验,核心员工,职场潜规则, 团队合作,做事经验,普升经验,技术深耕经验,少走弯路;

  • 专属交流群:高手交流,开阔眼界,分享心得,每天成长,厚积薄发,任何技术问题,任何疑难杂症,都可以咨询,讨论,交流!

深入理解系列

  • 深入理解计算机系统

  • 深入理解操作系统(调度,内存,网络,IO)

  • 深入理解并发技术全景指南

  • 深入理解编程语言

  • 深入理解算法与数据结构

  • 深入理解网络协议

  • 深入理解网络编程

  • 深入理解性能优化

  • 深入理解分布式技术

  • 深入理解数据库

  • 深入理解代码设计

  • 深入理解架构设计

详细了解:极客星球  ,现在加入超级优惠,早点加入一起学习成长,早点突破成长瓶颈:

ff72132faa5ec8b3ff54bd9a7ee7e5e5.png

扫码加入,加我微信:fr35331508,拉你进VIP群。

求点赞关注,在看,分享一键三连0e3ee9c432b4a3861a7a406ff1779958.png

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值