阅读更多

27顶
0踩

编程语言

转载新闻 用 Unix 的设计思想来应对多变的需求

2012-05-04 10:18 by 正式编辑 nemohq 评论(14) 有11869人浏览
在软件设计领域,有很多设计模式可以参考。但本文作者另辟蹊径,从UNIX设计思想的角度来阐述什么是好的软件设计,什么样的设计可以把需求变更对软件开发的影响降到最低。

以下为原文文章

我曾经在《UNIX传奇》的下篇中写过一些关于UNIX的设计哲学和思想(这里重点推荐大家看一下《The Art of UNIX Programming》),以前也写过一篇《一些软件设计的原则》。不过,这些内容都太多了,记不住。其实,这么多年来,我的经验告诉我,无论是UNIX设计,还是面向对象设计,还是别的什么如SOA、ECB、消息、事件、MVC、网络七层模型、数据库设计等等,他们都在干三件事——解耦,解耦,还是解耦!所谓解耦,就是让软件的模块和模块间尽量少地依赖起来


一、现实当中的例子

让我先举几个现实生活中的例子:

1、现实社会中,制造灯具的工厂完全不关心制造灯泡的工厂,制造灯泡的工厂也完全不关心制造灯具的工厂。但是,灯泡和灯饰可以很完美地组合成用户所喜欢的样子。他们是怎么做到的?

2、互联网上,网站开发者完全不用关心用户在用什么样的操作系统,什么样的客户端浏览器 (当然事实上,浏览器的不标准也让网站开发者很头痛,这里只是举个例子)。反过来,用户也不关心网站开发者在用什么技术来开发网站。但是大家在完全不关心对方的情况下,却可以很正常地协同工作在一起。为什么?

这样的例子太多了。为什么可以做成这样呢?因为大家依赖的是同一个接口。灯具和灯饰并不互相依赖,它们依赖的是一个接口;网站开发者和网站的用户依赖的还是接口——HTTP协议。这就是面向对象的核心思想——依赖于接口而不是依赖实现,这也就是解耦。当你看过这两个例子以后,我希望你以后设计的软件至少不能比我们现实社会中的这些方法要差。不然,你就是在让社会倒退了。

你可能会说,这与UNIX和应对需求变化有什么关系?下面让我们再来看一下UNIX的设计。

二、UNIX设计的例子

下面是几个UNIX下的例子:

1、UNIX下,所有的硬件都可以通过文件的方式存取。于是,软件和硬件的耦合被解开了,操作系统只需要把硬件统统变成文件即可。而程序也只需要使用三个工具:一个是fd,一个是read(),一个是write(),就可以来操作任意的硬件了,这就是抽象,简单到不行。

2、UNIX下,所有的命令都可以用管道串起来(管道绝对是个伟大的发明),这样,所有的命令间的交互全部解耦到只依赖于STD_IN和STD_OUT设备上。最酷的是,用户可以使用管道任意地拼装那些命令,以完成各式各样的功能。管道这个设计思想可以映射为今天的Web Service,你可以任意地拼装各种Web Service。

看到这里,你会发现,这还是解耦,本质上来说,也是一种依赖倒置——OOD的精髓。但是,UNIX还不仅仅是这些。我们再来看几个例子:

1、UNIX下,软件都是绿色地安装。在iOS上更明显——各个程序间基本上互不干扰,这个程序产生的垃圾文件不会影响到另一个程序。你删除一个程序不会影响到另一个程序。你也可以删除这些程序,只要把内核留着,系统依旧可以启动。

2、UNIX下,你可以通过设置一些环境变量,让多种环境同时存在。例如,某个LAMP 用的是Apache 2.0、Mysql 4.0、PHP 4.0,而另一个个LAMP用的是Apache 2.2、Mysql 5.0、PHP 5.3,你不但可以方便地在系统中切换这两个环境,你甚至还可以同时启动它们。

3、UNIX下,你可以随意地替换你想要的程序。例如,你不喜欢bash,你可以替换成ksh/csh;你不喜欢awk,你可以替换成gawk;所有的东西都像零件一样,你不喜欢什么,你就可以替换什么。

这三个例子告诉了我们——当你把你的软件设计的耦合度非常低时,你就可以随意地组合,随意地安排你的系统。相当的灵活,灵活到Windows到今天都学不会。

三、应对需求变化

看到这里,你可能明白我想说的是什么了。你可能开始觉得怎么样的系统设计会更有效了。如果你还记得《Steve Y 对平台的长篇大论》,你就会知道我想说什么了。是的,我想说的就是,当你真正了解了UNIX的设计思想后,你会觉得今天的这些设计理念都是对UNIX设计思想的一种传承或是变种。这些设计理念就是:

  • 解耦、解耦、解耦。尽量地让你的模块不要在实现上耦合,而是耦合某个规范,某个标准
  • KISS、KISS、KISS。要做到高度解耦,你的模块就一定要很简单。当然不是说简单到只有几行代码,而是简单到只做一件事,并把这件事做到极致,然后通过某个标准拼装起来
  • 拼装、拼装、拼装。我想不起来是谁曾说过这样一句话:“当我想用一个模块的时候,我直接调用就好了,没有必要像C或Java一样,还要编译”。是的,拼装需要一个框架,需要一种标准协议,然后让所有的系统都耦合在这种规范上,各自独立运行。就像一个机器上的各个部件,当我觉得这个部件性能不好,换一个部件就是了(例如,当我们在尝试不同的算法的时候)。
你还可以再想想建材和家具市场,无论用户过来想要装修什么,我都可以满足用户的不同需求。只要要求是和家具装修相关,我基本上都能满足,不是吗?无论你怎么变,只要不变态,我基本上都可以满足你。这就是解耦、拼装带来的好处。

你可能会说我说得太简单了,或者,你可能觉得有一些系统这样做没必要。不错,我承认这一点。不过,你可以有选择的或多或少地试试(其实,我相信你已经在不自觉的或多或少的在使用这种方式开发软件了)。

  • 大小: 35 KB
来自: 酷壳
27
0
评论 共 14 条 请登录后发表评论
14 楼 dohkoos 2012-12-13 16:07
引用
UNIX下,软件都是绿色地安装。在iOS上更明显——各个程序间基本上互不干扰,这个程序产生的垃圾文件不会影响到另一个程序。你删除一个程序不会影响到另一个程序。你也可以删除这些程序,只要把内核留着,系统依旧可以启动。


这个就是个大笑话。也许古老的unix-like系统是这样,但现在的unix-like安装个软件的话,你就会发现依赖另一个package,装那个package时又发现依赖另一个library,就一直这样,反正依赖比较严重
13 楼 shirne 2012-05-07 10:53
耦合度低有它的好处,也有弊端,就是效率的问题。
关键是在这之间找一个平衡点。
系统的模块我不太了解,就拿一个简单的脚本来说:
每个函数都实现一个独立的功能,需用到就支调用。
这叫低耦合。
但某些模块,是需要多种功能,循环调用,如果把功能代码嵌入到里面,就可以提高效率。
但换成调用到公共模块,带来的就是效率的损失。
12 楼 wentao_tang 2012-05-07 10:16
其实 主要是在设计 和 过度设计间取得一个很好的平衡,这个才是比较困难的。
11 楼 sp42 2012-05-06 21:33
不错 顶 学习!
10 楼 hardPass 2012-05-06 10:26
有道理。但同时应该注意另外一个词:“过度设计”
9 楼 amcucn 2012-05-06 08:58
说起来容易,但做起来却是另外一回事,通常这样的思想大家都知道,难的是如何建立接口,每个人的站的角度不一样,定义的接口也不一样,所以最终很难定义一套可供大家完美使用的实现方案。
8 楼 xiaoyao3857 2012-05-06 00:18
楼主思想理解得很透彻啊,学校老师虽然偶尔会讲,但说出来时,总又没有说到实质有影响的话或例子!
7 楼 helloqiner 2012-05-05 21:41
越是真理越是简单。同样的道理,不同的人理解不一样。楼主的这篇文章值得想成为砖家级别的码农好好领悟下。
6 楼 xieye 2012-05-05 13:17
引用
例如,某个LAMP 用的是Apache 2.0、Mysql 4.0、PHP 4.0,而另一个个LAMP用的是Apache 2.2、Mysql 5.0、PHP 5.3,你不但可以方便地在系统中切换这两个环境,你甚至还可以同时启动它们。


嗯嗯,我知道。
5 楼 mathgl 2012-05-04 23:20
beyondqinghua 写道
你还可以再想想建材和家具市场,无论用户过来想要装修什么,我都可以满足用户的不同需求。只要要求是和家具装修相关,我基本上都能满足,不是吗?无论你怎么变,只要不变态,我基本上都可以满足你。这就是解耦、拼装带来的好处。

但是很多需求就是很BT,你懂的。


再bt的需求,只要能实现都会被还原到代码。
4 楼 weng 2012-05-04 21:11
好文章,得领悟
3 楼 beyondqinghua 2012-05-04 18:10
你还可以再想想建材和家具市场,无论用户过来想要装修什么,我都可以满足用户的不同需求。只要要求是和家具装修相关,我基本上都能满足,不是吗?无论你怎么变,只要不变态,我基本上都可以满足你。这就是解耦、拼装带来的好处。

但是很多需求就是很BT,你懂的。
2 楼 myaniu 2012-05-04 13:05
自从熟悉linux(unix)以后,设计软件一直都是这么干的,甚至一个系统中,都会同时采用多种开发语言,每个进程只关注一个问题,每种语言只用来开发它擅长的部分。以后复用也很容易。
1 楼 x_root 2012-05-04 12:58
引用
其实,这么多年来,我的经验告诉我,无论是UNIX设计,还是面向对象设计,还是别的什么如SOA、ECB、消息、事件、MVC、网络七层模型、数据库设计等等,他们都在干三件事——解耦,解耦,还是解耦!所谓解耦,就是让软件的模块和模块间尽量少地依赖起来。
有道理!

发表评论

您还没有登录,请您登录后再发表评论

相关推荐

  • 用UNIX的设计思想来应对多变的需求

    今天我也想从Unix设计思想的角度来说说什么是好的软件设计,什么样的设计可以把需求变更对开发的影响降低。(注意:这并不能解决用户或是PM的无理需求,面对无理需求,需要仔细分析需求,而用技术的手段无法搞定这个...

  • 用Unix的设计思想来应对多变的需求

    无论是Unix设计,还是面向对象设计,还是别的什么如SOA,ECB,消息,事件,MVC,网络七层模型,数据库设计,等等,他们都在干三件事——解耦,解耦,还是解耦!所谓解耦,就是让软件的模块和模块间尽量少地依赖起来...

  • 外加热强制循环蒸发器装配图(CAD).rar

    外加热强制循环蒸发器装配图(CAD).rar

  • 数控车床纵向进给系统设计.zip

    数控车床纵向进给系统设计.zip

  • vault_side_off_ominous.png

    j

  • 爬虫 bangumi名称和评论数

    爬虫 bangumi名称和评论数

  • 基于SpringBoot的垃圾分类回收系统(源码+数据库+万字文档)526

    基于SpringBoot的垃圾分类回收系统,系统包含两种角色:管理员、用户主要功能如下。 【用户功能】 首页:浏览垃圾分类回收系统信息。 个人中心:管理个人信息,查看历史记录和订单状态。 运输管理:查看运输信息,垃圾回收的时间和地点。 公告管理:阅读系统发布的相关通知和公告。 垃圾回收管理:查看垃圾回收的信息,回收类型和进度。 垃圾出库申请管理:提交和查看垃圾出库申请的状态。 【管理员功能】 首页:查看垃圾分类回收系统。 个人中心:管理个人信息。 管理员管理:审核和管理注册管理员用户的信息。 用户管理:审核和管理注册用户的信息。 运输管理:监管和管理系统中的运输信息。 公告管理:发布、编辑和删除系统的通知和公告。 垃圾回收管理:监管和管理垃圾回收的信息。 垃圾出库申请管理:审批和管理用户提交的垃圾出库申请。 基础数据管理:管理系统的基础数据,运输类型、公告类型和垃圾回收类型。 二、项目技术 编程语言:Java 数据库:MySQL 项目管理工具:Maven 前端技术:Vue 后端技术:SpringBoot 三、运行环境 操作系统:Windows、macOS都可以 JDK版本:JDK1.8以上都可以 开发工具:IDEA、Ecplise、Myecplise都可以 数据库: MySQL5.7以上都可以 Maven:任意版本都可以

  • 这篇文章是台湾大学(NTU)计算机科学与信息工程系(CSIE)2021年秋季学期算法设计与分析课程的第一份作业(Homework#1)的具体要求和题目描述 以下是主要内容的总结:

    内容概要:本文档是台湾大学计算机科学与信息工程系2021年秋季学期《算法设计与分析》课程的第一次作业(Homework#1)。作业包含四道编程题和三道手写题,旨在考察学生对算法设计和分析的理解与应用能力。编程题涉及汉诺塔、数组计算、矩形点对、糖果分配等问题;手写题涵盖渐近符号证明、递归方程求解、幽灵腿游戏优化、不公平的卢卡斯问题等。文档详细描述了每个问题的具体要求、输入输出格式、测试用例以及评分标准。此外,还提供了编程技巧和注意事项,如避免延迟提交、正确引用资料、处理大输入文件等。 适合人群:具备一定编程基础的本科生或研究生,特别是修读过或正在修读算法设计与分析相关课程的学生。 使用场景及目标:①帮助学生巩固课堂所学的算法理论知识;②通过实际编程练习提高解决复杂问题的能力;③为后续更深入的学习和研究打下坚实的基础。 其他说明:此作业强调团队合作和个人独立思考相结合的重要性,鼓励学生在讨论后用自己的语言表达解决方案,并注明参考资料。对于编程题,特别提醒学生注意输入文件可能较大,建议采取适当的优化措施以确保程序运行效率。

  • 基于SpringBoot的铁路订票管理系统(源码+数据库+万字文档+ppt)528

    基于SpringBoot的铁路订票管理系统,系统包含两种角色:管理员、用户主要功能如下。 【用户功能】 首页:浏览铁路订票管理系统的主要信息。 火车信息:查看火车的相关信息,包括车次、出发地、目的地和票价等。 公告资讯:阅读系统发布的相关通知和资讯。 后台管理:进行系统首页、个人中心、车票预订管理、车票退票管理等操作。 个人中心:管理个人信息,查看订单历史记录等。 【管理员功能】 首页:查看铁路订票管理系统。 个人中心:修改密码、管理个人信息。 用户管理:审核和管理注册用户的信息。 火车类型管理:管理系统中的火车类型信息。 火车信息管理:监管和管理系统中的火车信息,添加、编辑、删除等。 车票预订管理:处理用户的车票预订请求。 车票退票管理:处理用户的车票退票请求。 系统管理:管理系统的基本设置,公告资讯、关于我们、系统简介和轮播图管理。 二、项目技术 编程语言:Java 数据库:MySQL 项目管理工具:Maven 前端技术:Vue 后端技术:SpringBoot 三、运行环境 操作系统:Windows、macOS都可以 JDK版本:JDK1.8以上都可以 开发工具:IDEA、Ecplise、Myecplise都可以 数据库: MySQL5.7以上都可以 Maven:任意版本都可以

  • 塑料架注射模具设计.rar

    塑料架注射模具设计.rar

  • 基于json文件数据驱动的的接口测试框架.zip

    基于json文件数据驱动的的接口测试框架

  • 铁丝缠绕包装机设计-缠绕盘设计.rar

    铁丝缠绕包装机设计-缠绕盘设计.rar

  • Linux操作系统及常用命令详解.zip

    linux

  • 圆柱体相贯线焊接专机工作台设计.rar

    圆柱体相贯线焊接专机工作台设计.rar

  • 硬币分拣机设计.rar

    硬币分拣机设计.rar

  • 【机器学习与数据挖掘】行业级机器学习软件开发经验与教训:从LIBSVM和LIBLINEAR看算法部署及软件设计挑战

    内容概要:本文探讨了开发行业级机器学习和数据挖掘软件的经验与教训,指出当前研究界与工业界之间的脱节问题。作者分享了开发LIBSVM和LIBLINEAR的经验,强调了用户需求的重要性。大多数用户并非机器学习专家,期望简单易用的工具来获得良好结果。文章还详细介绍了支持向量机(SVM)的实际应用案例,包括数据预处理(如特征缩放)、参数选择等步骤,并提出了为初学者设计的简易流程。此外,作者讨论了在设计机器学习软件时应考虑的功能选择、选项数量、性能优化与数值稳定性等问题,强调了软件开发与实验代码的区别以及鼓励研究人员参与高质量软件开发的重要性。 适合人群:对机器学习软件开发感兴趣的科研人员、工程师及从业者,尤其是那些希望了解如何将学术研究成果转化为实际可用工具的人士。 使用场景及目标:①帮助非机器学习专家的用户更好地理解和使用机器学习方法;②指导开发者在设计机器学习软件时考虑用户需求、功能选择、性能优化等方面的问题;③促进学术界与工业界之间的合作,推动高质量机器学习软件的发展。 其他说明:本文不仅提供了具体的开发经验和技巧,还呼吁建立激励机制,鼓励更多研究人员投入到机器学习软件的开发中,以解决当前存在的研究与应用脱节的问题。

  • pandas学习代码,jypyter格式

    一天入门pandas代码

  • joblib-0.12.0-py2.py3-none-any.whl

    该资源为joblib-0.12.0-py2.py3-none-any.whl,欢迎下载使用哦!

  • 深度学习基于PyTorch==2.6.0和Transformers==4.48.0的XTuner环境配置:AI模型开发与优化依赖库列表

    内容概要:本文档《xtuner_requirements.txt》列出了用于支持特定项目(可能是机器学习或深度学习项目)运行所需的所有Python包及其版本。其中不仅包括常见的数据处理和科学计算库如numpy、pandas,还包括了与深度学习密切相关的库如torch、transformers等。值得注意的是,文档中还特别指定了NVIDIA CUDA相关组件的具体版本,确保了GPU加速环境的一致性和兼容性。此外,文档中也包含了从GitHub直接安装的xtuner库,明确了具体的提交哈希值,保证了代码来源的精确性。 适合人群:对机器学习、深度学习领域有一定了解并需要搭建相应开发环境的研发人员,尤其是那些希望复现特定实验结果或基于已有模型进行二次开发的研究者和技术爱好者。 使用场景及目标:①帮助开发者快速搭建完整的开发环境,确保所有依赖项正确无误;②为研究人员提供一个稳定的实验平台,以便于重复实验和验证结果;③作为项目协作的基础,确保团队成员之间的环境一致性,减少因环境差异带来的问题。 阅读建议:由于该文档主要为技术性依赖列表,在阅读时应重点关注所需安装的库及其版本号,特别是CUDA相关组件和自定义库(如xtuner)的安装方式。对于非技术人员而言,可能需要额外查阅相关资料来理解各库的作用。同时,在实际操作过程中,建议按照文档中的顺序逐一安装依赖,避免版本冲突等问题的发生。

  • vault_side_on_ominous.png

    j

Global site tag (gtag.js) - Google Analytics