KISS 原则和 YAGNI原则

2024-04-11 02:20
文章标签 原则 kiss yagni

本文主要是介绍KISS 原则和 YAGNI原则,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

设计模式专栏:http://t.csdnimg.cn/3a25S

目录

1.引言

2.代码并非行数越少越简单

3.代码复杂不一定违反 KISS 原则

4.如何写出满足 KISS 原则的代码

5.YAGNI原则和 KISS 原则的区别


1.引言

        KISS原则的英文描述有3种版本:Keep It Simple and Supid、 keep It Short and Simple、Keep It Simple and Straightforward。其实,它们要表达的意思差不多,即“尽量保持简单”。

        KISS原则是一个“万金油”一样的设计原则,可以应用在诸多场合。它不仅经常用来指导软件开发,还经常用来指导系统设计、产品设计等,如冰箱、建筑和手机的设计等,本书讲解的是代码设计,因此,接下来,我们重点讲解如何在程序开发中应用KISS 原则。

        我们知道,代码的可读性和可维护性是衡量代码质量的两个重要标准。而KISS原则就是保持代码可读和可维护的重要手段。代码足够简单,也就意味着容易读懂,bug比较难影藏,即便出现 bug,修复也比较简单。

        不过,KISS原则只是告诉我们,要保持代码“简单”,但并没有讲什么样的代码才算得上“简单”,更没有给出明确的方法来指导如何开发“简单”的代码。因此,KISS 原则虽然简单但不太容易落地。

2.代码并非行数越少越简单

        在下面的示例代码中,我们使用3种方式实现同一功能:检查输入的字符串ipAddress是否是合法的P地址。一个合法的P地址由4个数字组成,并且通过“.”进行分隔。每个数字的取值范围是 0~255(第一个数字比较特殊,不允许为0)。对比下面3段代码,读者认为哪一段代码符合KISS 原则呢?

//第一种实现方式:使用正则表达式
public boolean isValidIpAddressVl(String ipAddress)
{if(StringUtils.isBlank(ipAddress)){return false;}String regex = "^(1\\d{2)|2[0-4]\\d125[0-5]|[1-9]\\dl[1-9])\\."+"(1\\d(2}12[0-4]\\d|25[0-5]|[1-9]\\dl\\d)\\."+"(1\\d{2)12[0-4]\\d125[0-5]1[1-9]\\dl\\d)\\."+"(1\\d{2)12[0-4]\\d125[0-5]1[1-9]\\dl\\d)$";return ipAddress.matches(regex);
}//第二种实现方式:使用现成的工具类
Public boolean isValidIpAddressV2(String ipAddress)
{    if (Stringutils.isBlank(ipAddress))return false;String[] ipUtits = StringUtils.split(ipAddress,'.');if(ipUnits.length!=4){return false;}for(int i=0;i<4;++i){int ipUnitIntValue;try{            ipUnitIntValue = Integer.parseInt(ipUnits[i]);catch(NumberFormatException e){return false;}it (ipUnitIntValue <0 || ipUnitIntValue > 255){return false;}if(i == 0 && ipUnitIntValue== 0){return false;}}return false;
}//第三种实现方式:不使用任何工具类
public boolean isValidIpAddressV3(String ipAddress)
{char[] ipChars = ipAddress.toCharArray();int length = ipChars.length;int ipUnitIntValue = -l;boolean isFirstUnit = true;int unitsCount=0;for(int i=0;i<length;++i){char c = ipChars[i];if(c==''){if(ipUnitIntValue < 0 || ipUnitIntValue > 255) return false; if(isFirstUnit  && ipUnitIntValue ==0)return false;if(isFirstUnit )isFirstUnit = false;ipUnitIntValue=-l;unitsCount++;continue;}if (c < '0' || c >'9'){return false;if(ipUnitIntValue=-1)ipUnitIntValue =0;ipUnitIntValue=ipUnitIntValue*10+(c-'0');}if (ipUnitIntValue<0 && ipUnitIntValue >255)return false;if(unitsCount !=3)return false;return true;
}

        第一种实现方式利用正则表达式,3行代码就解决了问题。第一种实现方式的代码行数最少,那么是否符合KISS原则呢?答案是否定的。虽然第一种实现方式的代码行数最少,看似简单,但使用了比较复杂的正则表达式,而想要写出完全没有bug的正则表达式是很有挑战性的。对于不熟悉正则表达式的人,看懂并维护含有正则表达式的代码是比较困难的。基于正则表达式的实现方式导致代码的可读性和可维护性变差,因此,从KISS原则的设计初衷(提代码的可读性和可维护性)来看,这种实现方式并不符合 KISS原则。

        第二种实现方式使用StringUtils类和 Integer 类提供的一些现成的工具函数来处理IP地址字符串。第三种实现方式不使用任何工具函数,而是通过逐一处理IP地址中的字符来判断是否合法。从代码行数上来说,第二种实现方式和第三种实现方式的代码行数差不多。但第三种实现方式比第二种实现方式更有难度,更容易产生bug。从可读性来说,第二种实现式的代码逻辑更清晰、更好理解。相比来说,第二种实现方式更“简单”,符合KISS 原则。 虽然第三种实现方式稍微复杂,但其性能要比第二种实现方式高一些。从性能的角度说,选择第三种实现方式是不是更好呢?在回答这个问题之前,我们先解释一下为什么第三种实现方式的性能更高一些。一般来说,工具类的功能是通用和全面的,因此,在代码实现面、需要兼容和处理更多的情况、执行效率就会受到影响。而第三种实现方式,完全是自己操作底层字符,只针对IP地址这一种输入格式,没有其他不必要的处理逻辑,因此,在执行率方面,这种类似定制化的处理代码肯定比通用的工具类高。

        尽管第三种实现方式的性能更高,但我们还是倾向于选择第二种实现方式,因为第三种实现方式上实际是过度优化。除非isValidIpAddress函数是影响系统性能的瓶颈代码,否则,这样优化的投入产出比并不高,反而增加了代码实现的难度、牺牲了代码的可读性,而性能上的提升并不明显。

3.代码复杂不一定违反 KISS 原则

        上文我们提到,代码并非行数越少越简单,因为还要考虑逻辑复杂度、实现难度和代码的可读性等。如果一段代码的逻辑复杂、实现难度大、可读性也不太好,是不是一定违反KISS原则呢?在回答这个问题之前,我们先来看下面这段代码(来自《数据结5之美》中 KMP 算法的代码实现)。

//P算法:a、b分别是主串和模式串,n、m分别是主串和模式串的长度
public static int kmp(char[]a, int n, char[]b, int m){int[] next= getNexts(b,m);int j= 0;for(int i=0;i<n; ++i){while(j>0 && a[i] != b[j]){j=next[j-1]+ 1;}if(a[i]== b[j]){++j;}if(j==m){return i-m +l;}}return -l;
}
private static int[] getNexts(char[]b, int m){int[] next = new int[m];next[0]=-1;int k=-1;for(int i=1;i<m; ++i){while(k!=-1 && b[k +1]!=b[i]){k= next[k];}if (b[k + 1] == b[i]){++k;}next[i] = k;}return next;
}  

        上面这段代码逻辑复杂、实现难度大和可读性差,但它并不违反KIS原则,KMP算法以高效著称,当需要处理长文本字符串匹配问题(如几百MB大小的文本内容的匹配),或者字符串匹配是某个产品的核心功能(如Vim、Word等文本编辑中的文本查找),抑或字符串匹配算法是系统性能瓶颈时,我们就应该选择KMP算法。而KMP算法本身具有逻辑复杂、实现难度大和可读性差特点,因此,使用复杂的算法解决复杂的问题,并不违反KISS原则。

        不过,平时的项目开发涉及的字符串匹配问题大多针对较小的文本,在这种情况下,直接调用编程语言提高的现成的字符串匹配函数即可。如果是KMP算法实现较小文本的字符串匹配,就违反KISS原则了。也就是说,对于同一段代码,在某个应用场景下满足KISS原则,换一个应用场景后可能就不满足 KISS 原则了。

4.如何写出满足 KISS 原则的代码

        关于如何写出满足 KISS 原则的代码,前面已经讲了一些方法,这里总结一下。

        1)慎重使用过于复杂的技术来实现代码,如复杂的正则表达式、编程语言中过于高级的语法等。

        2)不要“重复造轮子”,首先考虑使用已有类库。根据作者的经验,如果自己实现类库那么产生 bug 的概率更高,维护成本也更高。

        3)不要过度优化。尽量避免使用一些“奇技淫巧”(如使用位运算代替算术运算、使用复杂的条件语句代替 if-else 等)来优化代码。

5.YAGNI原则和 KISS 原则的区别

        当YAGNI(You Ain’t Gonna Need It)原则用在软件开发时,其含义是: 不要去设计当前用不到的功能;不要去编写当前用不到的代码。实际上,这条原则的核心思想是:不要过度设计。和 KISS 原则一样,YAGNI原则也称得上“万金油”一样的设计原则。

        例如,某系统暂时只使用 Redis来存储配置信息,以后可能会用到ZooKeeper。根据 YAGNI原则,在未用到ZooKeeper之前,我们没必要提前编写这部分代码。当然,这并不是说就不需要考虑代码的扩展性了。我们还是有必要预留扩展点,在需要引入ZooKeeper时,能够在不改太多代码的情况下完成扩展。

        又如,不要在项目中提前引入不需要依赖的开发包。Java程序员经常使用Maven或Grade 管理项目依赖的类库,我们发现,有些程序员为了避免开发中类库的缺失而频繁地修改Maven或 Gradle 配置文件,提前向项目里引入大量常用的类库。实际上,这种做法违反YAGNI原则。

        从刚才的分析可以看出,YAGNI原则与KISS原则并非一回事。KISS原则讲的是“如何做(尽量保持简单),而 YAGNI原则讲的是“要不要做”(当前不需要的,就不要做)。

这篇关于KISS 原则和 YAGNI原则的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/892829

相关文章

JVM内存调优原则及几种JVM内存调优方法

JVM内存调优原则及几种JVM内存调优方法 1、堆大小设置。 2、回收器选择。   1、在对JVM内存调优的时候不能只看操作系统级别Java进程所占用的内存,这个数值不能准确的反应堆内存的真实占用情况,因为GC过后这个值是不会变化的,因此内存调优的时候要更多地使用JDK提供的内存查看工具,比如JConsole和Java VisualVM。   2、对JVM内存的系统级的调优主要的目的是减少

水处理过滤器运行特性及选择原则浅谈

过滤属于流体的净化过程中不可缺的处理环节,主要用于去除流体中的颗粒物或其他悬浮物。水处理过滤器的原理是利用有孔介质,从流体中去除污染物,使流体达到所需的洁净度水平。         水处理过滤器的滤壁是有一定厚度的,也就是说过滤器材具有深度,以“弯曲通 道”的形式对去除污染物起到了辅助作用。过滤器是除去液体中少量固体颗粒的设备,当流体进入置有一定规格滤网的滤筒后,其杂质被阻挡,而

重写equals和hashCode的原则规范

当符合以下条件时不需要重写equals方法:     1.     一个类的每一个实例本质上都是唯一的。     2.     不关心一个类是否提供了“逻辑相等”的测试功能     3.     超类已经改写了equals方法,并且从超类继承过来的行为对于子类也是合适的。     4.     一个类时私有的或者是package私有的,并且可以确定它的equals方法永远不会被调用。(这

职场关系课:职场上的基本原则(安全原则、进步原则、收益原则、逃生舱原则)

文章目录 引言安全原则进步原则收益原则逃生舱原则 引言 职场上的王者,身体里都应该有三个灵魂: 一个文臣,谨小慎微,考虑风险; 一个武将,积极努力,谋求胜利; 一个商人,精打细算,心中有数。 安全原则 工作安全:保住自己的工作和位置信用安全:保住个人的信用,如果领导看到了你的信用受损,你和领导的关系可能会持续恶化。人身安全:有的时候你会遇到偏执的人,要及时和

浅谈数据库、JVM、缓存、SQL等性能调优方法和原则

浅谈数据库、JVM、缓存、SQL等性能调优方法和原则 java互联网架构 2019-07-07 13:19:00 性能优化基本是BAT等一线互联网公司程序员必备的技能,以下为大家完整揭晓性能完整的优化方案和方法:包含web网站调优、数据库、JVM调优、架构调优等方案。 第一:Web网站调优 1、尽可能减少HTTP请求:图片合并 (css sprites),Js脚本文件合并、css文件

软件架构设计的七大原则(附架构资料)

软件架构设计的七大原则(附架构资料) 程序媛菲儿 2019-06-24 19:05:29 一、开闭原则 开闭原则(Open-Closed Principle OCP)是指一个软件实体,如类、模块和函数应该对扩展开放,对修改关闭。所谓的开始,是用抽象构建框架,用实现扩展细节。可以提高软件系统的可维护性和可复用性。开闭原则是面向对象中最基础的原则,实现开闭原则的基本思想就是面向抽象编程。 以某

设计模式六大原则:单一职责原则 + 依赖倒置原则

感悟二:   "站在不同的高度, 看到不同的风景"吧.       正如老总看的是公司发展方向, 主管却在看业绩, 经理在看项目, 小弟们在看代码... 感悟三: 设计模式很重要     设计模式是我到公司才接触的事物, 主要是讲述一种面向接口的编程思维, 按照设计模式所编写的代码, 会比学校那种直接实现功能的代码繁琐一点, 增加很多看似多余的虚类或者接口. 但是这种代码更加具有拓

一般软件开发流程和一般软件的测试原则

一般软件开发的流程: 软件测试的原则:

单一职责原则 SRP

单一职责原则,就一个类而言,引起其变化的原因只应该有一个。本质上是实现程序松耦合的目的,当功能改变的时候对其他功能尽可能少的影响。

开闭原则 OCP

开闭原则,当业务逻辑需要改动时,在不修改原来代码的情况下,进行功能扩展