重构手法——提炼函数、搬移函数、以多态取代条件表达式

目录

    • 我的心路历程
    • 我的学习概括
      • Extract Method(提炼函数)
        • 动机*--做法
        • 动机--做法*
      • Move Method(搬移函数)
        • 动机*--做法
        • 动机--做法*
      • Replace Conditional with Polymorphism(以多态取代条件表达式)
        • 动机*--做法
        • 动机--做法*

我的心路历程

今天开始学习《重构改善既有代码的设计》这本书。我首先翻阅了前言部分,书中写了什么是重构。什么是重构?

我的思考心路:
对于重构的认识,我认为程序开发并不是一个设计后即动作的行为,它需要在开发过程中持续整理程序,最大限度减少引入错误的概率的过程。

继续翻阅前言,书中写了本书有什么。告诉了书本的章节对应内容。第1章是一个重构的Demo,第2章是重构的原则定义,第3章是“坏代码”的味道,第5-12章是核心部分——重构的一些手法,等等。从哪开始快速入手看呢?

我的思考心路:
翻开第1章,我了解到这个Demo主要是个影片出租店程序,程序的功能是计算顾客消费金额并打印详单。翻看了第1章目录:起点–重构第一步–分解并重组statement()–运用多态取代与价格相关的逻辑条件–结语。我看看结语讲了什么?

翻阅结语,结语中表示第一章的Demo示范了数个重构手法。重构后的程序使责任分配更合理,这种风格,将迥异于过程化风格,也使码农难以再满足于结构化风格。看看Demo示范了那几个重构手法?

Demo中重构手法有:
P110 提炼函数
P142 搬移函数
P255 以多态取代条件表达式
P171 自封装字段
P227 以state/strategy取代类型码

我的学习概括

Extract Method(提炼函数)

动机*–做法

我自己的理解如下:
概念:
这个手法做法是,把可以被组织到一起的代码提炼到一个独立函数中,并用名称来表示用途。

应用场景:

  1. 过长的函数。
  2. 需要注释才能理解用途的代码。

达到目的:
得到简短而命名良好的函数。

优势:

  1. 函数粒度小,复用机会大,且覆写更容易。
  2. 使高层函数读起来像一系列注释。

动机–做法*

我自己的理解如下:
做法简图:
在这里插入图片描述

结合上图,我的关注点如下:

copy过程中注意安全:

  1. 源函数层面:注意考虑变量、参数的作用域
  2. 提炼出的代码层面:临时变量问题

意外1:
源函数中变量、参数的作用域与新函数不一致。
措施:若是局部变量,新函数里传入这部分参数。

意外2:
copy过程中临时变量被修改了。
措施:将被提炼代码段处理为一个查询,将结果赋值给相关变量。

Move Method(搬移函数)

动机*–做法

我自己的理解如下:

概念简图:

在这里插入图片描述

概念:
类A中函数Fun a与类B有调用关系的情况下,Fun a最常引用的类B中建立类似形为Fun a`的新函数,则Fun a变成单纯的委托函数,或可完全移除Fun a。

应用场景:

  1. 一个类有太多行为。
  2. 或类与类之间有太多合作而形成高度耦合。

达到目的:
使系统中的类更简单,系统任务交付更干脆。

动机–做法*

我自己的理解如下:
做法简图:
在这里插入图片描述

结合上图,我的关注点如下:
移动路径方向:函数Fun a与对象B交流比自己对象A交流多,所以搬移方向往B合适。

搬移过程中注意安全:

  1. 源类A的子类超类层面:若源类A的子类和超类,有函数Fun a的声明不能进行搬移。除非类B表现出多态性

意外1:
新目标类B中缺少对A的引用机制。
措施:把源对象类A的引用当作参数,传给新目标类B的Fun a`。

意外2:
Fun a包含异常处理,搬移后出现异常处理异常。
措施:根据逻辑判断某个异常该由哪个类处理,若类A负责,则异常处理留在类A。

Replace Conditional with Polymorphism(以多态取代条件表达式)

动机*–做法

我自己的理解如下:

概念简图:

在这里插入图片描述

概念:
将条件表达式的每个分支放进一个子类的覆写函数中,然后将原始函数声明为抽象函数。

应用场景:
同一组表达式在程序多个地方出现。

优势:

  1. 减少编写明显的条件表达式。
  2. 使“类型码switch语句”、“基于类型名称的if-then-else语句”出现减少。
  3. 避免条件表达式情况下,添加新类型时要更新所有表达式的缺点。
  4. 降低系统各部分间的依赖。

动机–做法*

我自己的理解如下:

结合上面的概念简图,我的关注点如下:
替换过程中注意安全:

  1. 多态性层面:必须有一个继承结构。它可以有2种选择来做到这个结构。一般情况下采取以子类取代类型码的方法。若出现对象已经创建后,要修改类型码;或者,要重构的类已经有子类,则采取以state/strategy取代类型码的方法。
  2. 超类层面:超类中某些private可能需要声明为protected。另外,超类中容纳条件表达式的函数要声明为抽象函数。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/549222.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

FTP服务器架设详细图解

FTP是File Transfer Protocol(文件传输协议)的缩写,用来在两台计算机之间互相传送文件。FTP服务作为Internet最古老的服务之一,无论在过去还是现在都有着不可替代的作用。在企业中,对于一些大文件的共享,通…

gradle下bug修正后问题仍存在解决思路

目录我的学习过程我的学习心路热加载配置bug问题总结我的学习过程 前天写的client-server聊天项目写完后,今天进行了调试。我用到的是out目录下的server.class文件和client.class文件。 先后启动两个命令行窗口来进行测试的。 使用java server启动服务端窗口。 再使…

IP-tools

IP-tools 网管员的第三只眼^ Ip-tools是一款功能齐全的网管软件,可以随时随地的向网管员报告网络的运行情况ip-tools自身集成多种tcp/ip使用工具,如本地信息、链接信息、端口扫描、ping、WHOIS、finger、nslookup、telnet、NetBIOS等功能。界面是全英的&…

用git提交代码到远程仓库遇到的问题

目录我的学习过程git环境配置(Mac版)git原理图git的push操作思路遇到的问题我的学习过程 昨天重写了一遍聊天程序,准备提交到git上进行代码管理。结果遇到了不少问题。我照着网上的教程进行操作,一步一步踩了很多坑。 git环境配…

数字示波器的激烈竞争

计算机、通信以及消费类电子产业的快速发展成为示波器发展的不竭动力&#xff1b;厂商不断从技术上对示波器进行改进更使其发展日新月异。 <?xml:namespace prefix o ns "urn:schemas-microsoft-com:office:office" />数字示波器自它诞生的第一天起&#xf…

git pull和push整理和归纳

目录各个模块概念工作区版本库暂存区远程仓库pull和push流程相关的命令暂存区相关版本库相关远程仓库相关利用远程仓库协作开发各个模块概念 我的理解&#xff1a; Git是版本管理工具&#xff0c;它主要对指定目录下的一些特定的文件的修改进行版本管理。 相关的模块有&#x…

重装vcenter后恢复原来制作的模板!

重新安装vcenter后发现原来用vcenter创建的模板没有了。清单中只显示现有的4台虚拟机&#xff0c;没有显示模板。其实找回来也很简单&#xff01;在清单中找到数据存储&#xff0c;在相应的模板文件夹中找到的.vmtx文件添加到清单中去即可&#xff01;转载于:https://blog.51ct…

Linux中点号,星号,加号,问号实战

目录Linux中的星号&#xff0c;点号和加号概念我的学习过程&#xff1a;我的思考过程&#xff1a;点号&#xff08;.&#xff09;星号&#xff08;*&#xff09;加号&#xff08;&#xff09;问号&#xff08;?&#xff09;linux星号&#xff0c;点号&#xff0c;加号&#xf…

如何调整HOOK的跳转指令

可以按这样的方式来存放 长度A 长度A 用于调整Short JMP 用于存放一些信息 |调整后的原HOOK代码 |原始代码(HOOK) |临时LONG JMP区| 信息区| 1). 调整…

广播地址的计算方法(与运算、或运算)

目录我的学习过程Python中逻辑运算符notandor位运算符取反&#xff08;~&#xff09;与&#xff08;&&#xff09;或&#xff08;|&#xff09;广播地址计算方法IP地址子网掩码网络地址广播地址广播地址计算举例我的学习过程 今天学习UDP的单播、多播、广播中&#xff0c;…

Wt::WTreeNode

2019独角兽企业重金招聘Python工程师标准>>> A single node in a tree. 〔 这个 widget 渲染的是一棵树的一个节点。〕 A tree list is constructed by combining several tree node objects in a tree hierarchy, by passing the parent tree node as the last arg…

匿名内部类探究——它是一个实例

目录我的学习过程匿名内部类概述匿名内部类探究代码验证&#xff08;匿名内部类是一个实例&#xff09;结论我的学习过程 昨天想学习一下Java8新特性&#xff0c;看到Lambda表达式可以替代匿名内部类。我对匿名内部类不太理解&#xff0c;决定学习一下。并进行了下面的归纳和思…

利用SQL查找表中的质数(prime number)和完全数(perfect number)以及几个有趣的SQL语句...

之前在某次interview中被老外问到如何用SQL找出列上的质数和完全数的问题&#xff1b;我当时已经多年没有写过这种考算法和SQL技巧(纯粹的技巧)的语句了&#xff0c;乍遇此问题倒是有些棘手。现在录以记之&#xff0c;供人参考. SQL> create table numbers(NO int) ;表已创建…

Lambda表达式及应用

目录Lambda表达式概念应用在forEach()方法使用用来替代匿名内部类代码验证&#xff08;Lambda表达式替代匿名内部类&#xff09;Lambda表达式 概念 语法形式&#xff1a; () -> {} 组成&#xff1a; 括号&#xff1a;表示参数列表&#xff1b;箭头&#xff1a;表示lambda…

UrlRewriter 伪url的配置

UrlRewriter 是微软封装好了的一个URL重写组件。使用它可以让我节约很多自已开发的时间。 好了&#xff0c;开始讲述我的应用经验&#xff0c;这只是很菜鸟的经验&#xff0c;高手就不用看了。 第一步&#xff0c;请从此下载此组件。解压&#xff0c;把UrlRewriter.dll copy到你…

网络地址和广播地址的快速计算方法

目录前提条件方法原理网络地址快速计算示例广播地址快速计算示例前提条件 由IP地址和子网掩码&#xff0c;快速计算网络地址和广播地址。 小窍门前提&#xff1a;当子网掩码组成只有255和0组成时。 方法原理 利用255&#xff08;或者0&#xff09;和其他数字的&&#xf…

职场提醒:面试失败n次以后

投了一份简历&#xff0c;是中国一个很有名的医药公司的职位。面试后对方回复说&#xff0c;我不符合他们的要求。之前也参加过很多公司面试&#xff0c;结果都是说我经验不足。一次次的失败也让我对自己有了新的认识&#xff0c;也不断的刺激我。发现一个工作经验不足的人&…

关于list遍历时sychronizedList方法和synchronized同步块的线程安全问题思考

目录我的难点sychronizedList方法无法解决List遍历时线程不安全问题sychronizd同步块为什么可以解决List遍历时线程不安全问题我的思考过程我的难点 sychronizedList方法无法解决List遍历时线程不安全问题 关于慕课网的socket课程学习中&#xff0c;qiujuer老师在课程中讲到&…

WinCE CEDDK之DMA相关函数

CEDDK提供了DMA的相关函数&#xff0c;在CEDDK/DDK_DMA/ddk_dma.c中定义。实际上里面最有用的就两个函数&#xff0c;HalAllocateCommonBuffer(..)和HalFreeCommonBuffer(..)分别用于为DMA申请和释放 首先介绍一下会用到的DMA适配器结构&#xff0c;在ceddk.h中定义,如下: type…