最近写代码的一些心得

一、不放过细节


很多的隐藏很深的bug就是代码里面一些小细节不注意导致的,比如一个变量被定义的位置,或者一个自己不了解细节的api调用等。很多细节的处理也决定了代码质量,比如命名、代码规范的处理等。
我们看下面这段代码:

private void reduceValueConvert(CouponCategoryDTO dto) {
    BigDecimal reduceValue = BigDecimal.ZERO;
    if (MarketConstant.CategoryType.ZKQ.equals(dto.getType()) || MarketConstant.CategoryType.SPQ.equals(dto.getType())) {
        reduceValue = dto.getReduceValue().multiply(new BigDecimal(10)).subtract(new BigDecimal(10)).abs();
    }
    if (MarketConstant.CategoryType.SPQ.equals(dto.getType()) && MarketConstant.DiscountType.FPQ.equals(dto.getDiscountType())) {
        reduceValue = dto.getReduceValue();
    }
    dto.setReduceValue(reduceValue);
}

这是我前段时间在排查问题的时候看到的代码,刚好这段代码的细节问题非常多。这里先简单介绍一下这个函数的目的:根据传入的CouponCategoryDTO判断优惠券是商品券还是折扣券,如果是商品券MarketConstant.CategoryType.SPQ,则不做转换,如果是MarketConstant.CategoryType.ZKQ,则将传入的dto里面的reduceValue设置成供页面显示的X折这种形式。逻辑非常简单,但代码看上去却差点意思,我们来剖析一下看看:

命名

这个方法名叫reduceValueConvert,从字面意思上理解的话,可以理解为“转换减少值”,应该是针对传入的dto对象中的reduceValue做转换的。但是具体要做什么样的转换,以及为什么要转换,单从方法名上是看不出来的,那假设换一种命名方式叫做:getDisplayReduceValue,这时候理解起来就知道,ok,这是因为reduceValue需要转换成ui需要的显示格式,所以定义一个函数专门用来干这件事情,方法名具体了很多,也更好理解了。

传参

我们看之前这个方法传了个CouponCategoryDTO进去,虽然说这样传参减少了参数的数量,但是一旦传入这个对象,这就是个引用传递,那么在方法内部是可以修改这个参数内部属性的数据的(这个函数确实这么干了😁),对外部调用来说,如果后续依赖这个参数对象的话,不建议直接传对象进方法,而是new一个新的对象,或者只传需要的参数进入这个方法就可以了。

简洁

我们看到,函数内部有2个if判断,而后面那个if判断会修改前面那个if代码块里面的reduceValue返回值,阅读起来还是比较啰嗦的,有简化空间;对于静态枚举变量,可以使用static import简化很多代码。

BUG

这个代码里面是隐藏一个bug的,细心就会发现,函数里面的2个if块,可能都没有满足,这种情况下,函数会将dto里面的reduceValue设置成BigDecimal.ZERO,这里就引发了bug了。产生这个BUG的原因其实还是写代码在细节处理上的坏习惯:不要在函数内部改变参数对象的属性

重构后代码

这里我对这段代码进行了一些重构,先看代码:

private BigDecimal getDisplayReduceValue(BigDecimal originalReduceValue, Integer categoryType, Integer discountType) {
    if (ZKQ.equals(categoryType) || SPQ.equals(categoryType)) {
        if (!FPQ.equals(discountType)) {
            return originalReduceValue.multiply(BigDecimal.TEN).subtract(BigDecimal.TEN).abs();
        }
    }
    return originalReduceValue;
}

我们看到,重构后的代码,函数名非常具体,就是去获取显示折扣,然后传入的3个参数名也很具体,使用了static import让代码相对简洁一些。只在指定的if块里面进行转换,其它情况仍然返回originalReduceValue。代码没有改变任何参数的值,具体对返回值做什么处理,由调用该函数的使用者来决定。代码看上去也比之前更容易维护了。

二、认真学习设计原则SOLID


SOLID原则是5大设计原则的首字母简称,分别为:

单一职责原则 SRP

在任何一个软件模块中,应该有且只有一个被修改的原因。这里强调2个点:
1、职责要单一,其它地方动了,不应该影响我,我动了,也不应该影响别人。
2、不能有多于一个职责,因为一旦职责多了,一个改动会影响另一个。
举个实际场景的例子,比如我有一个类,是叫OrderService,那这里面应该都是和订单相关的函数,如果万一出现一个支付的、优惠券的、购物车的,那就破坏了单一职责原则(其实这个例子不好,因为订单服务职责也太多了,应该拆解为订单查询,下单,订单支付等)。

开闭原则 OCP

软件实体应该对扩展开放,对修改关闭。这里也强调2点:

  • 有新需求时,可以对现有代码进行修改,以适应新的变化;
  • 类一旦设计完成,就可以独立进行工作,不要再对其做任何修改。

不修改就意味着不影响现有业务,也就不会引发BUG。所以我们在设计类时,要考虑怎么样既可以实现扩展功能,又不需要修改代码。
这一点我体会很深刻,我现在公司的一个项目由于已经上线了一年左右了,迭代了无数版本,业务逻辑已经比较复杂了,大家现在写代码有点像是在修水管,拧上一处阀门,往往导致另外一处地方漏水了。。最后很多精力花在救火上面,导致生产效率越来越低。
这里有几个设计模式我推荐大家学习一下,可以让代码避免过早陷入复杂性:

  • 装饰者模式:Wrap一个新类来扩展功能
  • 策略模式:制定一个策略接口,让不同的策略实现成为可能
  • 适配器模式:不改变原有类的基础上适配新功能
  • 观察者模式:灵活添加和删除观察者(Listener)来扩展系统功能

里氏代换原则 LSP

程序中的父类都应该可以正确地被子类替换。我发现很多程序员写代码的时候,其实不太擅长使用继承和抽象关系。其实面向对象设计里面最伟大的概念就是抽象,理解了抽象,才能对现实世界的业务进行建模,才能化繁为简,设计出简洁的软件架构。
在进行抽象设计的时候,程序中不应该出现instanceof关键词,因为这种设计破坏了LSP原则,将导致父类无法被复用(因为只有在特定子类时才生效)。子类中使用的函数应该在父类中被定义,子类和父类在行为表现上一定要一致。

接口隔离原则 ISP

多个特定场景的接口,要好过一个宽泛的通用接口。
1、不要强迫用户依赖那些他们并不使用的接口
2、使用多个专门的接口比使用一个总接口要好
这个比较好理解,就是我们现在都喜欢写一个XXXService,然后在里面定义一大堆的函数,这样的写法是违背ISP原则的,因为一个接口里面定义了太多的依赖函数,其实我们在其他地方调用该接口时,可能只依赖其中某几个函数,但是却要引入一个很大的依赖关系,如果修改了其中某个东西,很容易导致其他地方出错。所以尽量分成多个接口来开发。比如查询、修改分成2个接口来开发。

依赖倒转原则 DIP

模块之间交互应该依赖于抽象,而不是具体实现。
1、大家依赖的是一个约定,而不关心具体实现细节
2、领域层不应该依赖基础设施层
这2点非常重要,我们在写代码时,经常会涉及到调用第三方模块,比如我订单服务可能会查询商品服务,那么我们之间应该首先建立好一个抽象接口约定,当我调用的时候,我只需要调用这个抽象接口就可以了,不需要关心其他服务是用什么框架、甚至编程语言实现的。这里面Java里面的Spring框架是通过依赖注入方式来实现,还有类似FeignClient这种也是一个很好的方式。然后领域层(业务逻辑层)不应该依赖基础设施层(框架、中间件等),意思也是说,我们不应该让自己的业务代码被框架侵入太深,否则一旦这个框架有问题不满足需要,那我们就比较被动了。

总结

最近写代码比较多,遇到很多问题,也总结了很多经验,以上只是一部分。
PS:上周又遇到一个重大线上事故,我自己复盘下来,收获很大,回头抽时间写出来跟大家分享,避免踩坑。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容