需求变更,代码改的像辣鸡 - 论代码质量

· 浏览次数 : 55

小编点评

这篇文章是一位程序员在处理需求变更时,对自己代码质量的反思和对技术书籍阅读体验的分享。文章通过一个具体的代码修改案例,展示了如何在不改变业务逻辑的前提下,通过重构提高代码质量。同时,作者结合自己的阅读经历和技术书籍的学习,谈到了对代码质量的看法。 1. **紧急需求与代码挑战**: - 描述了一个紧急需求变更,要求增加品牌数据。 - 分析了代码的修改历程,从原始版本到最终的整洁版本。 - 感叹于作者能够迅速适应需求变化并解决问题。 2. **代码质量与整洁之道**: - 引入了《代码整洁之道》一书的观点,强调代码质量的衡量标准。 - 通过书中的例子,说明了如何通过重构提升代码质量。 - 分享了阅读该书后的感悟,以及对技术书籍重要性的认识。 3. **技术书籍的阅读体验**: - 叙述了作者在阅读技术书籍时的体验和收获。 - 结合个人经历,讨论了技术书籍对于提升编程能力的作用。 - 提到了当前社会技术书籍阅读趋势的变化。 4. **面试中的技术书籍影响**: - 分享了作者在面试过程中如何被问及技术相关问题。 - 反映了技术书籍阅读对面试准备的影响。 - 通过自己的经验,说明了持续学习的重要性。 总的来说,文章通过一个实际案例,展示了如何通过重构代码来应对需求变更,同时也反映了作者对技术书籍阅读的态度和体验。它鼓励读者在技术实践中不断学习和成长,以提升自己的编程技能。

正文

 

一句注释引发的思考

接到一个有鸡毛信般的紧急需求(当然,002的需求向来是如此紧急的):大屏展示原来只有二个品牌数据,现增加到三个品牌的数据。一句话的需求,且没有业务逻辑变更,我认为可以迅雷不及掩耳之势,2小时收拾干净交差。当我满腔激情的定位的核心逻辑部分时,这样一句注释(见下图),让我顿时思绪天马行空:

这个作者经历了什么样一个撕心裂肺的过程?但是可以肯点的是这一定是一个有想法的作者,不由得心中肃然起敬。

这段代码经历了多少次的蹂躏,才会让作者的心潮有如此波澜?

抑或,这到底提出了怎么样一个需求,让作者需要通过这样的注释来宣泄心中怨气?

注释图

   

  巴拉开代码修改记录,作者已经去别的地方高就了,要不是留了这些代码,实在想不起有这样的一个同事存在过;代码提交记录比较整洁,大部分代码是在5月29号提交,5月30大概是修复bug提交了小部分代码。如此看来,代码没有经历过什么苦难,这里的需求仅仅是每个品牌的门店按订单数量排序(如下图),想想怎么也闹出什么大动静... 再细读作者留下的代码,只能说作者给自己设置了难度系数(这说法太含蓄了),稍微有一点改动,便是牵一发而动全身,于是留了这样一个不太成熟且不太有价值的注释,想起了最近在读的一本书,Bob大叔的《代码整洁之道》,颇有一点不成熟的感触。

 

 

如何衡量代码质量

《代码整洁之道》一书开篇第一句话就是一个著名的论断:衡量代码质量的唯一有效标准:WTF/min。 一看到WTF这样的简称就不明觉厉,一顿搜索居然没有找到“标准”的解释,更觉得高深莫测。 也是在后来一个偶然的时间看到原来是 What-The-Fuck 的缩写,看重看这张经典的图,一下子恍然顿悟:原来如此,就该如此。

原本学得这是一本好书,甚至觉得大部分程序都应该去阅读这类的书籍一遍(比如 马丁·福勒出品的《重构-改善既有代码的设计》)。 看到这样的论述,更觉此书接地气,仿佛与大师拉近了距离了。回想起过往种种,以及最近修改历史代码时的反应,没什么比WTF更有表达力了。

 

 再回到开篇讲的注释,当时的作者必然也是有类似的反应,只是他的吐嘲对象是他自己,或者他只会认为这是需求的而不是代码的问题。所以 WTF/min作为衡量代码质量的唯一有效标准,还得加一个定语:优秀的Coder喊出的WTF次数,才是真正的标准。 至于如何为优秀的代码,在《代码整洁之道》,《重构-改善既有代码的设计》等经典书籍里都有详细的描述:

  • 从变量命名,到注释,到方法,到类,到模块都有非常详细的规范;
  • 从抽象到边界,到依赖也有完整的方法论;
  • 从SOLID设计原则到,组件相关的 CRP,CCP,CRP等原则都有从理论到落地的解说。

 坦白讲,读完这些书后,我感觉自己原来写的代码,很多都缺乏这样的思考,也有不少代码相当丑陋,甚至觉得:在code这件事让,只能算得上初窥门径。于是越时读书,越觉得自己无知。最近读技术书籍的间歇,顺便翻读了几本名人传记:奥本海默为了让人觉得他是天才,总是"偷偷"的读书学习;特斯拉甚至在病危之际也是一心读书;让我们有在剑手的钱学森利用所有空闲时间阅读方能一年完成硕士学位,并获得航空和数学双博士,成本最年轻的终生教授...(省略好多荣誉)。他们无一不是通过自己努力读书,思考,实践终成我等学习的楷模。最近招聘的经历中,大部分应聘的人几乎不看技术书籍的了,也是让我捉摸不透。

读技术书籍没用了吗

最近两月一直忙于面试,沟通了没有100个也有80个候选人,大部分人都没有了读书习惯,更不用说技术书籍了,倒是有部分说觉得blog比书籍有用多了。有些说工作太忙,有得说没有用....他们中有的在传统公司朝9晚5,有的在互联网企业996,有从小企业过来的,也有从阿里,快手过来的... 其中一个人的话,让我记忆非常深刻,最后环节,我问他现在还读书不,他说为了进阿里,他非常努力了2年,把《深入理解java虚拟机》读了2遍,《高性能mysql》,《深入理解kafka》... 都仔细阅读了,后来进了阿里,觉得这些书都没有没啥用了,也不在阅读其他技术书籍了,最后我不知道应该说啥,毕竟我没去过阿里,于是结束了面试。就我自己而言,早些年面试也是被问得体无完肤,也有过这样的心态阅读了大量类似 《深入理解java虚拟机》,《MySQL技术内幕:InnoDB存储引擎》,《RocketMQ技术内幕:RocketMQ架构设计与实现原理》,《从Paxos到Zookeeper:分布式一致性原理与实践》等书籍,确实也让自己在后来的面试中可以从容面对。但越是阅读,越是感觉自己不知道的东西越多,越是想要通过阅读来充实自己。于是又开始阅读系统设计,架构一类的书籍。时至今日,读到《代码整洁之道》时,依然觉得即使在做了10年的编码这件事上,不懂的依然有非常之多(哈哈,也许是悟性不够)。 从前面的名人,到我身边认识的人,大凡优秀的人的,独有阅读的习惯,并且有大量阅读自己专业相关的书籍。

最近和媳妇探讨读书这件事儿,说我周末在家除了溜娃就是刷新闻,现如今的新闻包括热搜又大部分是没有“营养”的,也聊到目下短视频盛行,下到3岁,上到70,地铁上,公园里,城市里,老家里... 到处都是,当然也包括我们自己的爸妈,谈及此,心中不觉升起一股名族忧心(哈哈,操心有点多了)。于是我放下了新闻,当然了周5晚上,等娃睡着后,我们买些宵夜,找一部金典电影还是保持着。 一段时间后,发现一个周阅读10几个小时好像也挺正常的,阅读成生活的一部分。也没有了过去那种读了多久了,要休息下的,看看新闻,看看电视的想法了。现在想来,读书也好,新闻,短视频也罢,本质且没啥差异,内心富足就好。

再回到前面的面试,也不知道,我在面试过程中,把读书这一块看得如此重,是否合适,但是我相信:喜欢阅读技术书籍的人,应该都不会太差。 

回到前面的代码

回家开篇的注释问题,想和大家一直分享下代码重构过程,如果不幸被作者看到,希望不要介怀,就如Bob大叔所讲,每个程序员都应该接专业眼光的检视(哈哈也许我也不是那么专业)。 需求比较简单,就是两个品牌下的门店根据订单数排序。 现在的需求是增加了第三个品牌,门店信息有品牌属性。

如果作者阅读了Clean Code ,他就会明白代码走向整洁的4原则:

  • 运行所有测试;
  • 不可重复;
  • 表达了程序员的意图;
  • 尽可能减少类和方法的数量。

就会把排序算法抽离出来,与业务逻辑分离,避免大量重复;

如果他深刻喊出了 Don't Repeat YourSelf, 就不会有么多 ConsultationOrderRank 对象的出事化,甚至不会单独处理有数据与没数据的情况。

如果作者阅读了Clean Architecture,他就会明白要面向抽象,而不是具体去编程,

他就会面向品牌这个概念去编程,而不是面向具体的品牌1,品牌2去实现。

 


//需求变更,改的像辣鸡。
if (CollectionUtils.isEmpty(orderList)) {
            List<CfgStore> allStoreList = cfgStoreService.getStoresBLAndBabyBL();
            List<CfgStore> bellaList = allStoreList.stream().filter(st -> {
                return st.getType() == 0;
            }).sorted(Comparator.comparingInt(CfgStore::getStoreId)).collect(Collectors.toList());

            ArrayList<ConsultationOrderRank> ballaResult = new ArrayList<>();
            int bellaIndex = 0;
            for (CfgStore store : bellaList) {
                ConsultationOrderRank consultationOrderRank = new ConsultationOrderRank();
                consultationOrderRank.setStoreName(store.getNameAlias());
                consultationOrderRank.setStoreId(store.getStoreId());
                consultationOrderRank.setOrderNum(0);
                consultationOrderRank.setSort(bellaIndex);
                ballaResult.add(consultationOrderRank);
                bellaIndex++;
            }
            List<ConsultationOrderRank> blRankResult = ballaResult.stream()
                    .sorted(Comparator.comparing(ConsultationOrderRank::getSort)).collect(Collectors.toList());

            List<CfgStore> babyBellaList = storeList.stream().filter(st -> {
                return st.getType() == 1;
            }).sorted(Comparator.comparingInt(CfgStore::getStoreId)).collect(Collectors.toList());

            ArrayList<ConsultationOrderRank> babyBallaResult = new ArrayList<>();
            int babyIndex = 0;
            for (CfgStore store : babyBellaList) {
                ConsultationOrderRank consultationOrderRank = new ConsultationOrderRank();
                consultationOrderRank.setStoreName(store.getNameAlias());
                consultationOrderRank.setStoreId(store.getStoreId());
                consultationOrderRank.setOrderNum(0);
                consultationOrderRank.setSort(babyIndex);
                babyBallaResult.add(consultationOrderRank);
                babyIndex++;
            }

            List<ConsultationOrderRank> babyRankResult = babyBallaResult.stream()
                    .sorted(Comparator.comparing(ConsultationOrderRank::getSort))
                    .collect(Collectors.toList());
            Order order = Order.builder().consultationOrderRankStBellaList(blRankResult)
                    .consultationOrderRankBabyBellaList(babyRankResult).build();

            return order;
        }

        List<CfgStore> others = storeList.stream().filter(store -> {
            return !Arrays.stream(storeIdArr).collect(Collectors.toList()).contains(store.getStoreId());
        }).collect(Collectors.toList());

        Map<Integer, CfgStore> storeMap = storeList.stream().collect(Collectors.toMap(CfgStore::getStoreId, store -> {
            return store;
        }));

        //品牌1门店ID
        List<Integer> blIdList = storeList.stream().filter(st -> st.getType().equals(0))
                .map(CfgStore::getStoreId).collect(Collectors.toList());
        //品牌2门店ID
        List<Integer> babyblIdList = storeList.stream().filter(st -> st.getType().equals(1))
                .map(CfgStore::getStoreId).collect(Collectors.toList());

        //品牌2分组数据
        Map<Integer, List<HeOrder>> babyblMap = orderList.stream()
                .filter(order -> babyblIdList.contains(order.getStoreId()))
                .collect(Collectors.groupingBy(HeOrder::getStoreId));
        //品牌2分组数据
        Map<Integer, List<HeOrder>> blMap = orderList.stream()
                .filter(order -> blIdList.contains(order.getStoreId()))
                .collect(Collectors.groupingBy(HeOrder::getStoreId));

        //品牌1排行数据
        List<ConsultationOrderRank> bellaList = new ArrayList<>();
        //品牌2排行数据
        List<ConsultationOrderRank> babyBellaList = new ArrayList<>();
        //品牌1
        for (Entry<Integer, List<HeOrder>> entry : babyblMap.entrySet()) {
            CfgStore cfgStore = storeMap.get(entry.getKey());
            String storeName = cfgStore.getNameAlias();
            if (Strings.isNotBlank(storeName)) {
                List<HeOrder> orderNum = entry.getValue();
                ConsultationOrderRank consultationOrderRank = new ConsultationOrderRank();
                consultationOrderRank.setStoreName(storeName);
                consultationOrderRank.setStoreId(entry.getKey());
                consultationOrderRank.setOrderNum(orderNum == null ? 0 : orderNum.size());
                babyBellaList.add(consultationOrderRank);
            }
        }

        List<CfgStore> otherbabyBlList = others.stream().filter(store -> {
            return store.getType() == 1;
        }).collect(Collectors.toList());

        for (CfgStore store : otherbabyBlList) {
            ConsultationOrderRank consultationOrderRank = new ConsultationOrderRank();
            consultationOrderRank.setStoreName(store.getNameAlias());
            consultationOrderRank.setStoreId(store.getStoreId());
            consultationOrderRank.setOrderNum(0);
            babyBellaList.add(consultationOrderRank);
        }

        //品牌2
        for (Entry<Integer, List<HeOrder>> entry : blMap.entrySet()) {
            CfgStore cfgStore = storeMap.get(entry.getKey());
            String storeName = cfgStore.getNameAlias();
            if (Strings.isNotBlank(storeName)) {
                List<HeOrder> orderNum = entry.getValue();
                ConsultationOrderRank consultationOrderRank = new ConsultationOrderRank();
                consultationOrderRank.setStoreName(storeName);
                consultationOrderRank.setStoreId(entry.getKey());
                consultationOrderRank.setOrderNum(orderNum == null ? 0 : orderNum.size());
                bellaList.add(consultationOrderRank);
            }
        }

        List<CfgStore> otherBellaList = others.stream().filter(store -> {
            return store.getType() == 0;
        }).collect(Collectors.toList());

        for (CfgStore store : otherBellaList) {
            ConsultationOrderRank consultationOrderRank = new ConsultationOrderRank();
            consultationOrderRank.setStoreName(store.getNameAlias());
            consultationOrderRank.setStoreId(store.getStoreId());
            consultationOrderRank.setOrderNum(0);
            bellaList.add(consultationOrderRank);
        }
        //品牌1排序
        List<ConsultationOrderRank> blRank = bellaList.stream().sorted(Comparator.comparing(ConsultationOrderRank::getOrderNum).reversed())
                .collect(Collectors.toList());
        int blSort = 0;
        int blIndexCounter = 0;
        for (int i = 0; i < blRank.size(); i++) {
            //订单=0, 订单值不同, 递增
            boolean flag = blRank.get(i).getOrderNum() == 0 || (i != 0 && blRank.get(i).getOrderNum() != blRank.get(i - 1).getOrderNum());
            if (flag) {
                blSort = blSort + blIndexCounter + 1;
                blIndexCounter = 0;
            } else {
                if (i != 0) {
                    blIndexCounter++;
                }
            }
            blRank.get(i).setSort(blSort);
        }

        //品牌2排序
        List<ConsultationOrderRank> babyBlRank = babyBellaList.stream().sorted(Comparator.comparing(ConsultationOrderRank::getOrderNum).reversed())
                .collect(Collectors.toList());
        int babySort = 0;
        int babyIndexCounter = 0;
        for (int i = 0; i < babyBlRank.size(); i++) {
            //订单=0, 订单值不同, 递增
            boolean flag = babyBlRank.get(i).getOrderNum() == 0 || (i != 0 && babyBlRank.get(i).getOrderNum() != babyBlRank.get(i - 1).getOrderNum());
            if (flag) {
                babySort = babySort + babyIndexCounter + 1;
                babyIndexCounter = 0;
            } else {
                if (i != 0) {
                    babyIndexCounter++;
                }
            }
            babyBlRank.get(i).setSort(babySort);
        }

 

 我相信作者如果经常阅读技术书籍,写出的代码应该是这样的。

        //统计每个品牌每个门店订单数量
        for (Integer brandType : brandTypeList){
            Map<Integer, Long> theBrandStoreOrderCount = orderList.stream().filter(order -> brandType.longValue() == order.getBrandType()).collect(Collectors.groupingBy(HeOrder::getStoreId, Collectors.counting()));
            List<CfgStore> brandStores = storeList.stream().filter(store -> store.getType().equals(brandType)).collect(Collectors.toList());

            List<ConsultationOrderRank> storeOrderRank = new ArrayList<>();

            brandStores.forEach(store ->{
                Long orderCount = 0L;
                if (theBrandStoreOrderCount.containsKey(store.getStoreId())){
                    orderCount = theBrandStoreOrderCount.get(store.getStoreId());
                }
                ConsultationOrderRank storeOrder = ConsultationOrderRank.builder()
                        .storeId(store.getStoreId())
                        .orderNum(orderCount.intValue())
                        .storeName(store.getStoreName())
                        .sort(0).build();
                storeOrderRank.add(storeOrder);
            });
            List<ConsultationOrderRank> sortedStoreRank = storeOrderRank.stream().sorted(Comparator.comparing(ConsultationOrderRank::getOrderNum).reversed()).collect(Collectors.toList());

            setSortWithSameRankNum(sortedStoreRank);

            BrandOrderStatistic statistic = BrandOrderStatistic.builder()
                    .name(CfgStoreEnum.getValueByCode(brandType))
                    .storeOrderRank(sortedStoreRank)
                    .brandType(brandType).build();
            brandOrderStatistics.add(statistic);
        }

 

如此这般,我们当时践行了编码里的童子军规:当你离开营地时候,要让它比你来的时候更整洁干净。

成为一名优秀的程序员!

 

与需求变更,代码改的像辣鸡 - 论代码质量相似的内容:

需求变更,代码改的像辣鸡 - 论代码质量

一句注释引发的思考 接到一个有鸡毛信般的紧急需求(当然,002的需求向来是如此紧急的):大屏展示原来只有二个品牌数据,现增加到三个品牌的数据。一句话的需求,且没有业务逻辑变更,我认为可以迅雷不及掩耳之势,2小时收拾干净交差。当我满腔激情的定位的核心逻辑部分时,这样一句注释(见下图),让我顿时思绪天马

上周热点回顾(7.1-7.7)

热点随笔: · 程序员失业日记1:工作五年,交接半天 (小码A梦)· 学习.NET 8 MiniApis入门 (tokengo)· C#/.NET/.NET Core优秀项目和框架2024年6月简报 (追逐时光者)· 需求变更,代码改的像辣鸡 - 论代码质量 (2J)· 如何找到并快速上手一个开源项

撮合前端平台在低代码平台的落地实践

基于传统认知,前端产品直接触达消费者,往往具有高度的定制化、需求变更频繁等特点,要求具有很好的动态性, 能够满足不同客户的需求。那么能否建设类似的前端中台产品,我们姑且称之为“前端领域产品”,实现接入团队端到端能力复用呢?我们在撮合业务线中进行了一系列思考和探索。

精准测试探索

什么是精准测试?通常研发提测的需求有代码变更,针对研发的代码变更点以及关联点进行测试,我们称之为精准测试。

我又学会了使用Range实现网络文件下载的断点续传

目录前言1、Range请求头1.1、概述1.2、使用限制1.3、范围请求1.4、预防资源变更2、断点续传下载实现2.1、流程设计2.2、代码实现2.3、运行结果3、RandomAccessFile4、思维拓展参考资料 前言 在某次摸鱼的过程中,老大突然后面冒出来说要做一个拉取文件到本地的需求(写的时

责任链和策略设计模式-基于Java编程语言

在日常代码的编写中,业务需求的变化总是不定的。文中描述的责任链和策略设计模式能有效满足代码编写的开闭原则,能更加有效的应对随时变化的业务需求。

Git 小技巧:忽略某些文件的更改

作为一枚合格的代码贡献者,时常需要跟踪自己或者团队代码的变更,那么就很有必要了解并掌握一些软件代码版本管理工具或者系统,比如 Git、SVN、CVS、VSS等。

6.0 Python 使用函数装饰器

装饰器可以使函数执行前和执行后分别执行其他的附加功能,这种在代码运行期间动态增加功能的方式,称之为`"装饰器"(Decorator)`,装饰器的功能非常强大,装饰器一般接受一个函数对象作为参数,以对其进行增强,相当于C++中的构造函数,与析构函数。装饰器本质上是一个python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象.它经常用于有迫切需求的场

实践指南-前端性能提升 270%

当我们疲于开发一个接一个的需求时,很容易忘记去关注网站的性能,到了某一个节点,猛地发现,随着越来越多代码的堆积,网站变得越来越慢。本文就是从这样的一个背景出发,着手优化网站的前端性能,并总结出一套开发习惯,让我们在日常开发时,也保持高性能,而不是又一次回过头来优化性能。

规则引擎调研及初步使用

生产过程中,线上的业务规则内嵌在系统的各处代码中,每次策略的调整都需要更新线上系统,进行从需求->设计->编码->测试->上线这种长周期的流程,满足不了业务规则的快速变化以及低成本的更新试错迭代。因此需要有一种解决方案将商业决策逻辑和应用开发者的技术决策分离开,在系统运行时能去更新管理业务规则。