关闭

财富坊cff888: 阿里JAVA开发手册零度的思考理解(一)

标签: java阿里java开发手册
4502人阅读 评论(15) 收藏 举报
分类:

转载请注明原创出处,谢谢!

缘由

阿里JAVA开发手册已经发表有很长时间了,值得认真研究思考推广

  • 阿里官方的Java代码规范标准,这份开发手册不仅规范了一些开发细节,也提出了很多工程开发的哲学,值得好好阅读。

  • 可谓包罗万象,几乎日常Java开发中方方面面都有所涉及。

  • 每一条都是前人踩过的坑,通过血的教训总结出来的。

  • 能公布出来真是造福全部Java开发者。

  • 开发手册详细列举如何开发更加高效,更加容错,更加有协作性,力求知其然,更知其不然,结合正反例,提高代码质量。比如,异常日志处理时的各种不规范行为;集合转换的各种坑;创建线程池出现的等待队列OOM等。

的确阿里JAVA开发手册值得我们好好阅读和思考,每一条都是前人踩过的坑,通过血的教训总结出来的。所以今天就其中一点自己的思考理解进行分享。

阿里JAVA开发手册

看完这条,个人觉得主要是圈复杂度,由于代码是人写的,并且需要人来进行维护,如果足够的复杂的话,那么编写出现错误的可能性都很大,并且维护理解起来难度也非常高,以及后期如果需要扩展本来就很复杂再加一个很简单的功能都变得很困难(相信大家一定都有这样的经历)。

圈复杂度

圈复杂度(Cyclomatic complexity)是一种代码复杂度的衡量标准。
在软件测试的概念里,圈复杂度用来衡量一个模块判定结构的复杂程度,数量上表现为独立线性路径条数,即合理的预防错误所需测试的最少路径条数。圈复杂度大说明程序代码可能质量低且难于测试和维护,根据经验,程序的可能错误和高的圈复杂度有着很大关系。

看看上面阿里JAVA开发手册里面提到的,如果非得使用if()…else if()…else…方式表达逻辑,【强制】避免后续代码维护困难,请匆超过3层。如果超过3层的if-else的逻辑判断代码可以使用卫语句、策略模式、状态模式等来实现。

其实在我看来,使用卫语句、策略模式、状态模式就是来降低圈复杂度,让代码更加简单,这样不管是编写代码人员以及维护人员都可以非常方便了解到本质意思。

虽然阿里JAVA开发手册提到的是if()…else if()…else…方式表达逻辑,延伸下,关于多次嵌套循环等道理也一样,需要考虑优化的。

思路分析

判断树

如上图,其实看起来就是一颗树结构,相对来说其实比较复杂了,优化的思路其实就是把树结构变成顺序结构即可,那样条理就清晰了,总体思路是这样的,下面看看使用卫语句、策略模式、状态模式怎么达到的。

卫语句

卫语句?卫语句就是把复杂的条件表达式拆分成多个条件表达式,比如一个很复杂的表达式,嵌套了好几层的if - then-else语句,转换为多个if语句,实现它的逻辑,这多条的if语句就是卫语句。

其中卫语句示例如下:

public void today() {
    if (isBusy()) {
        System.out.println("change time.");
        return;
    }
    if (isFree()) {
        System.out.println("go to travel."); 
        return;
    }
    System.out.println("stay at home to learn Alibaba Java Coding Guidelines.");
    return;
}

其实这个比较简单,每一个if对应叶子节点的一条路径(每个if基本就return了)。

策略模式

概述:使用这个模式来将一组算法封装成一系列对象。通过传递这些对象可以灵活的改变程序的功能。

策略模式比较有名的就是诸葛亮的三个锦囊妙计说起,如图:

诸葛亮为什么要这么麻烦,做三个锦囊?他完全可以只做一个锦囊,将这三个妙计都写在它上面。可他没有这么做,而是正确的运用了策略模式做了三个锦囊。这样做的好处十分明显诸葛亮一个锦囊写一个妙计,他的思路十分清晰,不会三个计策相互混乱。赵云看妙计的时候也十分方便,什么时候看哪个妙计,使用十分方便,如果三个妙计混在一起,他就没这么方便了。

在JDK中java.util.Comparator#compare()就是使用的策略模式,比如我们经常对商品进行排序,条件有很多啊,按照商品浏览量、价格、更新时间、【价格、时间】、【浏览量、更新时间】(进行升序、降序操作)其实这个也是上面那颗树,需要做的就是每次取其中一条叶子节点。很多时候这些判断都是写在一个公用的方法里面,进行大量的判断之后写排序,而JDK怎么做的呢?把变化的比较判断拿出来,其实判断树中每个叶子结点就是一种策略,想象我们平时怎么做的呢? 都是把Comparator#compare()写好(可能有很多实现Comparator接口的排序算法)每次我们调用的时候选择其中一种即可。

与卫语句不同的是,卫语句把每一个if对应叶子节点的一条路径。而策略模式是所以叶子都在实现Comparator接口了,具体开始用那个是调用的直接用(所以不会像卫语句那样看见很多if了)

状态模式

概述:当一个对象的内在状态改变时允许改变其行为,这个对象看起来像是改变了其类。主要解决的是对象的行为依赖于它的状态(属性),并且可以根据它的状态改变而改变它的相关行为。

  1. 封装了转换规则。
  2. 枚举可能的状态,在枚举状态之前需要确定状态种类。
  3. 将所有与某个状态有关的行为放到一个类中,并且可以方便地增加新的状态,只需要改变对象状态即可改变对象的行为。
  4. 允许状态转换逻辑与状态对象合成一体,而不是某一个巨大的条件语句块。
  5. 可以让多个环境对象共享一个状态对象,从而减少系统中对象的个数。

由于状态模式是封装了转换规则,所以一般树的深度最少需要2层以及上,个人理解的感觉就是一个流程了,比如*水低于0度是冰的状态–> 大于0度又变成液态–> 100度又变成沸腾的状态

状态模式与策略模式很像,策略模式是外驱动,而状态模式是内驱动。本质也是把判断树里面只取其中一条叶子的路径。

状态模式有一个明显的缺点:状态模式对”开闭原则”的支持并不太好,对于可以切换状态的状态模式,增加新的状态类需要修改那些负责状态转换的源代码,否则无法切换到新增状态,而且修改某个状态类的行为也需修改对应类的源代码。

程序员笑话

老婆给当程序员的老公打电话:下班顺路买十个包子,如果看到卖西瓜的,买一个。当晚老公手捧一个包子进了家门…老婆怒道:你怎么只买一个包子?!老公甚恐,喃喃道:因为我真看到卖西瓜的了。”

如果使用策略模式就简单了,2条信息,1:买十个包子。2:买一个西瓜,没有就算了。
虽然是个笑话,但是顺序的就是比判断的要简单。

总结

  • 圈复杂度概念
  • 卫语句
  • 策略模式
  • 状态模式

汇总:本质就是把较深的判断树,使用的时候,就是把判断树结构变成顺序结构即可,就是给出每个叶子的路径而不需要看起来是一颗复杂的树结构。

思考

阿里JAVA开发手册

这是阿里JAVA开发手册其中一条明细,为什么呢,结果是啥,怎么出乎意料啦??期待你的留言和分析!!!


上面的一些分析都是个人自己的理解和思考,如果发现有不对的希望留言指出,谢谢!!!
如果读完觉得有收获的话,欢迎点赞加关注。


个人公众号

匠心零度公众号.jpg

9
4
查看评论

Java重构-策略模式、状态模式、卫语句

博文已搬迁:https://segmentfault.com/u/zeling/articles 前言 当代码中出现多重if-else语句或者switch语句时。弊端之一:如果这样的代码出现在多处,那么一旦出现需求变更,就需要把所有地方的if-else或者switch代码进行更改,要是遗漏了某一...
  • u014532217
  • u014532217
  • 2018-02-07 22:11
  • 120

《大话》之 策略模式 Vs 状态模式

一.简介: 策略模式: 背景: 商店要打折销售,各种版本的销售方式,让小菜心烦意乱 内容:    定义算法家族,分别封装起来,让他们之间可以户型替换,此模式让算法的变化,不会影响到使用算法的用户。 图文并茂:          ...
  • u013045878
  • u013045878
  • 2014-12-28 16:53
  • 860

卫语句,多层循环优化

【推荐】推荐尽量少用else, if-else的方式可以改写成: if(condition){ ... return obj; } // 接着写else的业务逻辑代码; 说明:如果非得使用if()...else if()...else...方式表达逻辑,【强制】请勿超过3层,超过请使用状态设计模式。...
  • lafengwnagzi
  • lafengwnagzi
  • 2017-02-21 17:47
  • 2251

卫语句

1.使用卫语句取代嵌套表达式 函数中的条件逻辑使人难以看清正常的执行途径。使用卫语句表现所有特殊情况。 动机:条件表达式通常有2种表现形式。第一:所有分支都属于正常行为。第二:条件表达式提供的答案中只有一种是正常行为,其他都是不常见的情况。    &#...
  • jw903
  • jw903
  • 2015-05-05 19:52
  • 11623

重构:switch语句改成策略模式还是状态模式

在重构篇里,可以用多态来取代switch语句,但是因为:一部影片可以在生命周期内修改自己的分类,一个对象却不能在生命周期内修改自己所属的类。所以这里不能用策略模式,用多态取代switch,而应该用状态模式(State)。public class Movie { public static f...
  • qq_21381465
  • qq_21381465
  • 2016-05-02 22:32
  • 2033

卫语句

阿里巴巴java开发手册 1.    表达异常的分支时,少用if-else方式,这种方式可以改写成: if (condition) {          ...
  • Paranoia_ZK
  • Paranoia_ZK
  • 2017-05-27 14:29
  • 559

一、(七)控制语句

| 来源:阿里技术 微信公众号(id:ali_tech)。仅供于交流、学习、研究,勿用于商业用途! 1. 【强制】在一个 switch 块内,每个 case 要么通过 break / return 等来终止,要么注释说明程 序将继续执行到哪一个 case 为止 ; 在一个 switch ...
  • qq_34803572
  • qq_34803572
  • 2017-10-01 19:38
  • 162

读书笔记

作者: 戈峰 欧永广 郑德立 张雁  1.  《实现模式》 I.实现模式价值观 Ø 沟通:代码是与人沟通的桥梁 Ø 简单:通常简单和沟通都是不可分割的 Ø 灵活性:灵活性的提高可能以复杂性的提高为代价 ...
  • njpjsoftdev
  • njpjsoftdev
  • 2016-08-05 16:38
  • 725

从这里开始,零度的追逐

每个人都是会改变的,以前的自己怎么也不会想到自己居然有一天居然会面对这样一个自己。   而从这里开始,零度的追逐。   也许对于现在的我来说确实是兴趣不在这里吧!但是既然未来无法改变了,就这样面对吧!既然你的专业是这个这个方向的,就没必要去欺骗自己进行所谓“垂...
  • hcy2319964421
  • hcy2319964421
  • 2016-03-07 16:08
  • 82

状态模式&策略模式

状态模式,这个模式在android中的运用特别,比如同一个按钮,需要根据点击事件的不同执行不同的业务逻辑,比如新浪微博,他里面有匿名登陆和实名登陆 匿名登陆环境:点击底部导航我的按钮,会显示一个提醒登陆的页面 如果是账号登陆环境,之间显示的我的个人信息,也就是说不同状态下点击事件展示的不一样的画...
  • qq_21220745
  • qq_21220745
  • 2017-02-23 22:35
  • 135
    微信公众号:匠心零度
      分享技术干货与实践,欢迎大家关注.
    个人资料
    • 访问:362755次
    • 积分:3180
    • 等级:
    • 排名:第12810名
    • 原创:65篇
    • 转载:14篇
    • 译文:0篇
    • 评论:221条
    博客专栏
    统计