重构 改善既有代码设计---第三章 代码坏味道

11 篇文章 0 订阅

3.1 重复代码
代码重复会让整个类变得更大,影响代码阅读。

1.同个类:不同方法中多次出现重复的代码或者表达式时,可以使用“提炼方法”的方式将重复代码或表达式提炼到方法A中,所有使用到这段代码或者表达式的方法通过对A方法的调用实现功能
2.两个互为兄弟的类中含有相同的代码或者表达式:将重复代码提炼到指定方法A中,再将A方法推入到超类,后两个子类通过调用超类的方法A完成功能
3.不同的类中出现重复代码:应将该重复代码提炼到独立的类中,后通过使用新类中定义的方法实现功能

3.2 过长函数
代码越长,越难理解。
程序的解释能力、共享能力、选择能力,都是由小型函数支持的。
摘抄:
让小函数容易被理解的真正关键在于一个好名字,如果你能给函数起个好名字,读者可以通过名字了解函数的作用,根本不必去看其中写了什么,
积极地分解函数,我们遵循一条原则:每当感觉需要以注释来说明什么的时候,我们就把需要说明的东西写进一个独立的函数中,并以其用途命名。
关键不在于函数长度,而在于函数“做什么”和“如何做”之间的语义距离。
1.函数中有大量的参数和临时变量:
1)使用replace temp query(以查询取代临时变量)来消除这些临时元素;introduce parameter object(引入参数对象)和preserve whole object(保持对象完整)则可以将过长的参数变得简洁;如果还是存在大量的临时变量,也可以使用replace method with method object(以函数对象取代函数)

3.3 过大的类
一个类只做一类事情
3.4 过长参数列
过长的参数列使用时会对不上号,难以理解。
1.使用对象替换过长的参数列,需要的参数只需要从对象中获取即可,参数的增删改,不会对使用的函数造成大影响。

3.5 发散式变化
定义:某个类常常会因为不同的原因在不同的方向上发生变化
一个类只做一类事情
3.6霰弹式修改
某种变化,会对多个类中方法进行修改(不能大意,否则会漏掉需要修改的地方,导致系统功能错误)
将发生改变的方法移动到一个类中
3.7 代码耦合
A类中某函数在进行功能实现时,对B类中多个不同的函数进行调用,耦合度过高。
解决:将所有使用B类数据和函数的地方进行提炼,再将该提炼的函数移至B类,A类中方法只需调用一次即可。
3.8 数据泥团
多个类中存在同名同意的字段,将这些字段整理出来,分别放入不同的类中,然后给它们赋予合适的函数签名(根据语义和用途对其进行提炼到新类中,对象命名根据用途命名
3.9 少用switch或使用多态
3.10 对于类中无用的属性或者组件,应将其消除,否则会使系统更难理解和维护
3.11 过度耦合的消息链(一个对象请求另一个对象,再向后者请求另一个对象…)
3.12 同样的功能,不一样的函数签名,合并代码。
3.13 注释:写明自己为什么做某事,也可以加入一定的功能实现步骤。

	*减少重复代码;一个方法不需要太长;警惕临时变量对程序的污染;坚持一个类只做一类事;方法参数列表不应太长;降低代码耦合;参数过多时可以使用对象替换;精简代码,避免类中出现无用的参数或组件;注释不宜过多,但要描述清楚为什么写此函数,可以加入一些实现步骤进行描述。以上改变可以使代码阅读者更好的阅读和理解代码。*
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
重构,一言以蔽之,就是在不改变外部行为的前提下,有条不紊地改善代码。多年前,正是本书原版的出版,使重构终于从编程高手们的小圈子走出,成为众多普通程序员日常开发工作中不可或缺的一部分。本书也因此成为与《设计模式》齐名的经典著作,被译为中、德、俄、日等众多语言,在世界范围内畅销不衰。 本书凝聚了软件开发社区专家多年摸索而获得的宝贵经验,拥有不因时光流逝而磨灭的价值。今天,无论是重构本身,业界对重构的理解,还是开发工具对重构的支持力度,都与本书最初出版时不可同日而语,但书中所蕴涵的意味和精华,依然值得反复咀嚼,而且往往能够常读常新。 目录: 第1章 重构,第一个案例1 1.1 起点1 1.2 重构的第一步7 1.3 分解并重组statement()8 1.4 运用多态取代与价格相关的条件逻辑34 1.5 结语52 第2章 重构原则53 2.1 何谓重构53 2.2 为何重构55 2.3 何时重构57 2.4 怎么对经理说60 2.5 重构的难题62 2.6 重构设计66 2.7 重构与性能69 2.8 重构起源何处71 第3章 代码味道75 3.1 DuplicatedCode(重复代码)76 3.2 LongMethod(过长函数)76 3.3 LargeClass(过大的类)78 3.4 LongParameterList(过长参数列)78 3.5 DivergentChange(发散式变化)79 3.6 ShotgunSurgery(霰弹式修改)80 3.7 FeatureEnvy(依恋情结)80 3.8 DataClumps(数据泥团)81 3.9 PrimitiveObsession(基本类型偏执)81 3.10 SwitchStatements(switch惊悚现身)82 3.11 ParallelInheritanceHierarchies(平行继承体系)83 3.12 LazyClass(冗赘类)83 3.13 SpeculativeGenerality(夸夸其谈未来性)83 3.14 TemporaryField(令人迷惑的暂时字段)84 3.15 MessageChains(过度耦合的消息链)84 3.16 MiddleMan(中间人)85 3.17 InappropriateIntimacy(狎昵关系)85 3.18 AlternativeClasseswithDifferentInterfaces(异曲同工的类)85 3.19 IncompleteLibraryClass(不完美的库类)86 3.20 DataClass(纯稚的数据类)86 3.21 RefusedBequest(被拒绝的遗赠)87 3.22 Comments(过多的注释)87 第4章 构筑测试体系89 4.1 自测试代码的价值89 4.2 JUnit测试框架91 4.3 添加更多测试97 第5章 重构列表103 5.1 重构的记录格式103 5.2 寻找引用点105 5.3 这些重构手法有多成熟106 第6章 重新组织函数109 6.1 ExtractMethod(提炼函数)110 6.2 InlineMethod(内联函数)117 6.3 InlineTemp(内联临时变量)119 6.4 ReplaceTempwithQuery(以查询取代临时变量)120 6.5 IntroduceExplainingVariable(引入解释性变量)124 6.6 SplitTemporaryVariable(分解临时变量)128 6.7 RemoveAssignmentstoParameters(移除对参数的赋值)131 6.8 ReplaceMethodwithMethodObject(以函数对象取代函数)135 6.9 SubstituteAlgorithm(替换算法)139 第7章 在对象之间搬移特性141 7.1 MoveMethod(搬移函数)142 7.2 MoveField(搬移字段)146 7.3 ExtractClass(提炼类)149 7.4 InlineClass(将类内联化)154 7.5 HideDelegate(隐藏“委托关系”)157 7.6 RemoveMiddleMan(移除中间人)160 7.7 IntroduceForeignMethod(引入外加函数)162 7.8 IntroduceLocalExtension(引入本地扩展)164 第8章 重新组织数据169 8.1 SelfEncapsulateField(自封装字段)171 8.2 ReplaceDataValuewithObject(以对象取代数据值)175 8.3 ChangeValuetoReference(将值对象改为引用对象)179 8.4 ChangeReferencetoValue(将引用对象改为值对象)183 8.5 ReplaceArraywithObject(以对象取代数组)186 8.6 DuplicateObservedData(复制“被监视数据”)189 8.7 ChangeUnidirectionalAssociationtoBidirectional(将单向关联改为双向关联)197 8.8 ChangeBidirectionalAssociationtoUnidirectional(将双向关联改为单向关联)200 8.9 ReplaceMagicNumberwithSymbolicConstant(以字面常量取代魔法数)204 8.10 EncapsulateField(封装字段)206 8.11 EncapsulateCollection(封装集合)208 8.12 ReplaceRecordwithDataClass(以数据类取代记录)217 8.13 ReplaceTypeCodewithClass(以类取代类型码)218 8.14 ReplaceTypeCodewithSubclasses(以子类取代类型码)223 8.15 ReplaceTypeCodewithState/Strategy(以State/Strategy取代类型码)227 8.16 ReplaceSubclasswithFields(以字段取代子类)232 第9章 简化条件表达式237 9.1 DecomposeConditional(分解条件表达式)238 9.2 ConsolidateConditionalExpression(合并条件表达式)240 9.3 ConsolidateDuplicateConditionalFragments(合并重复的条件片段)243 9.4 RemoveControlFlag(移除控制标记)245 9.5 ReplaceNestedConditionalwithGuardClauses(以卫语句取代嵌套条件表达式)250 9.6 ReplaceConditionalwithPolymorphism(以多态取代条件表达式)255 9.7 IntroduceNullObject(引入Null对象)260 9.8 IntroduceAssertion(引入断言)267 第10章 简化函数调用271 10.1 RenameMethod(函数改名)273 10.2 AddParameter(添加参数)275 10.3 RemoveParameter(移除参数)277 10.4 SeparateQueryfromModifier(将查询函数和修改函数分离)279 10.5 ParameterizeMethod(令函数携带参数)283 10.6 ReplaceParameterwithExplicitMethods(以明确函数取代参数)285 10.7 PreserveWholeObject(保持对象完整)288 10.8 ReplaceParameterwithMethods(以函数取代参数)292 10.9 IntroduceParameterObject(引入参数对象)295 10.10 RemoveSettingMethod(移除设值函数)300 10.11 HideMethod(隐藏函数)303 10.12 ReplaceConstructorwithFactoryMethod(以工厂函数取代构造函数)304 10.13 EncapsulateDowncast(封装向下转型)308 10.14 ReplaceErrorCodewithException(以异常取代错误码)310 10.15 ReplaceExceptionwithTest(以测试取代异常)315 第11章 处理概括关系319 11.1 PullUpField(字段上移)320 11.2 PullUpMethod(函数上移)322 11.3 PullUpConstructorBody(构造函数本体上移)325 11.4 PushDownMethod(函数下移)328 11.5 PushDownField(字段下移)329 11.6 ExtractSubclass(提炼子类)330 …… 第12章 大型重构359 第13章 重构,复用与现实379 第14章 重构工具401 第15章 总结409 参考书目413 要点列表417 索引419 ——《豆瓣读书》

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值