重构手法千变万化,如果你看过《重构_改善既有代码的设计》这本书一定会很有感触,此文我从这本书中挑选个人觉得比较有用的几点,一些我在工作中觉得比较实用的重构手法。
(01) - 方法提炼
有句话叫,当你为了给一段代码写注释的时候,你就能考虑把他提炼一下了。
来看一个实际例子,登录
配合idea的Refactor->Extract->Method
原始代码如下:
其中2和3.1提炼到了private方法, 3提炼到了别的类里,因为明显这是个公共方法
来看一个实际例子,登录
配合idea的Refactor->Extract->Method
原始代码如下:
01.
public
String login()
02.
{
03.
//错误信息
04.
String error =
""
;
05.
//1 显示验证码框
06.
flag =
"1"
;
07.
//2 检查用户名密码为空
08.
returnInfoBean =
new
ReturnInfoBean();
09.
if
(StringUtils.isBlank(custId))
10.
{
11.
error =
"用户名不能为空"
;
12.
}
13.
else
if
(StringUtils.isBlank(custPassword))
14.
{
15.
error =
"登录密码不能为空"
;
16.
}
17.
if
(StringUtils.isNotBlank(error))
18.
{
19.
returnInfoBean.setSuccess(
false
);
20.
returnInfoBean.setReturnMessage(error);
21.
return
"login"
;
22.
}
23.
24.
try
25.
{
26.
//3 调用恒生登录
27.
HsDsApi hsApi =
new
HsDsApi(
";P 003"
,
"integrate/login"
, LOGGER);
28.
hsApi.put(
"logtype"
, logtype);
29.
hsApi.put(
"lognumber"
, custId);
30.
hsApi.put(
"password"
, custPassword);
31.
if
(StringUtils.isNotBlank(certificateType)) hsApi.put(
"certificatetype"
, certificateType);
32.
HsResponse respx = hsApi.callHsApi();
33.
returnInfoBean.setSuccess(respx.isSuccess());
34.
if
(respx.isSuccess())
35.
{
36.
//3.1 保存用户基本信息存入session
37.
String sessionKey = respx.getString(
"sessionkey"
);
38.
ICTLoginBean lb =
new
CTLoginBeanImpl();
39.
lb.setUserInfo(getUserInfo(sessionKey));
40.
lb.setAccountUtil(getAccountUtil(sessionKey));
41.
request.getSession().setAttribute(ICTLoginBean.sessionID, lb);
42.
}
43.
else
44.
{
45.
//3.2 登陆失败返回错误代码并处理
46.
throw
new
RuntimeException(
"【"
+ respx.getCode() +
"】"
+ respx.getMessage());
47.
}
48.
}
catch
(Exception r)
49.
{
50.
//3.3 调用失败
51.
error =
"系统异常,请稍候再试"
;
52.
LOGGER.error(
"调用恒生登录失败"
, r);
53.
}
54.
55.
//4 是否登陆成功
56.
if
(StringUtils.isNotBlank(error))
57.
{
58.
returnInfoBean.setSuccess(
false
);
59.
returnInfoBean.setReturnMessage(error);
60.
return
"login"
;
61.
}
62.
else
63.
{
64.
return
"first"
;
65.
}
66.
}
01.
public
String login()
02.
{
03.
returnInfoBean =
new
ReturnInfoBean();
04.
returnInfoBean.setSuccess(
true
);
05.
//1 显示验证码框
06.
flag =
"1"
;
07.
08.
//2 检查用户名密码为空
09.
if
(!checkLogin())
return
"login"
;
10.
11.
try
12.
{
13.
//3 调用恒生登录
14.
HsResponse respx = HsCommonAPi.login(logtype, custId, custPassword, certificateType);
15.
if
(respx.isSuccess())
16.
{
17.
//3.1 保存用户基本信息存入session
18.
saveInfoToSession(respx.getString(
"sessionkey"
));
19.
}
20.
else
21.
{
22.
//3.2 登陆失败返回错误代码并处理
23.
throw
new
RuntimeException(
"【"
+ respx.getCode() +
"】"
+ respx.getMessage());
24.
}
25.
}
26.
catch
(Exception r)
27.
{
28.
//3.3 调用失败
29.
returnInfoBean.setSuccess(
false
);
30.
returnInfoBean.setReturnMessage(
"系统异常,请稍候再试"
);
31.
LOGGER.error(
"调用恒生登录失败"
, r);
32.
}
33.
34.
//4 是否登陆成功
35.
return
returnInfoBean.isSuccess()?
"first"
:
"login"
;
36.
}
(02) - 分解临时变量
一个变量变用在多个地方,而且每次使用含义都不同,这个时候可以考虑把他分解
重构后
重构后
(03) - 以查询取代临时变量
如果一个临时变量在方法里赋值有点复杂,而且可能还会被复用。可以提炼到方法里可以说是第1话的子范例
重构后
重构后
(04) - 引入解释性变量
不解释自己看例子,目的为了代码有更好地可读性
重构后
重构后
(05) - 使代码更简洁
这部分功能,欢迎大家投稿
重构后
1.
if
( flag ==
1
){
2.
return
true
;
3.
}
4.
else
{
5.
return
false
;
6.
}
1.
return
flag ==
1
;
(06) - 尽早结束非正常逻辑
1.
if
(taskList !=
null
&& !taskList.isEmpty()){
2.
3.
//正常逻辑代码
4.
5.
return
count;
6.
}
1.
int
count =
0
;
2.
if
(taskList ==
null
|| taskList.isEmpty()){
3.
return
0
;
4.
}
5.
6.
//正常逻辑代码
7.
8.
return
count;
(07) - 满足条件立刻跳出循环
01.
public
boolean
contain(
int
year, Month month,
int
day) {
02.
boolean
found =
false
;
03.
for
(IPolyDate date : dateList) {
04.
if
(date.same(year, month.getMonth(), day)) {
05.
found =
true
;
06.
break
;
07.
}
08.
}
09.
10.
return
found;
11.
}
01.
public
boolean
contain(
int
year, Month month,
int
day) {
02.
for
(IPolyDate date : dateList) {
03.
if
(date.same(year, month.getMonth(), day)) {
04.
return
true
;
05.
}
06.
}
07.
08.
return
false
;
09.
}
(08) - 使用数据构分离公共逻辑,避免重复(坚持DRY原则)
DRY原则:不要重复粘帖你自己的代码。Don't repeat yourself
重构后
01.
public
void
testGetIntPart()
throws
Exception {
02.
assertEquals(
"0"
, digitTransform.getIntPart(
"0.01"
);
03.
assertEquals(
"1"
, digitTransform.getIntPart(
"1.2"
);
04.
assertEquals(
"1234"
, digitTransform.getIntPart(
"1234"
);
05.
assertEquals(
"1"
, digitTransform.getIntPart(
"1.01"
);
06.
assertEquals(
"0"
, digitTransform.getIntPart(
"0.01"
);
07.
assertEquals(
"11111"
, digitTransform.getIntPart(
"11111"
);
08.
assertEquals(
"1000"
, digitTransform.getIntPart(
"1000.11"
);
09.
}
重构后
01.
public
void
testGetIntPart()
throws
Exception {
02.
String[][] cases =
new
String[][] { {
"0.01"
,
"0"
}, {
"1.2"
,
"1"
},
03.
{
"1234"
,
"1234"
}, {
"1.01"
,
"1"
}, {
"0"
,
"0"
},
04.
{
"11111"
,
"11111"
}, {
"1000.11"
,
"1000"
} };
05.
06.
for
(
int
i =
0
, len = cases.length; i < len; i++) {
07.
assertEquals(cases[
1
], digitTransform.getIntPart(cases[
0
]));
08.
}
09.
}
相关推荐
通过**提炼方法**、**提取类**、**方法上移**等技术,可以消除重复代码,提高代码复用性。 2. **过长的方法**:长方法往往意味着功能复杂,不易理解和维护。可以通过**提取方法**、**组合方法**、**以查询取代临时...
总结,本篇文档涉及了C#编程中的关键重构技巧,包括封装集合以限制集合操作,移动方法以优化代码组织,以及使用面向服务架构来减少系统间的耦合。这些都是提升代码质量,保证软件可维护性的有效策略。在实际开发中,...
在进行Rails版本升级时,重构代码是一个绝佳的机会。本文将深入探讨为何这是个好时机来进行重构,并提供一系列实用的方法帮助您优化代码。 #### 为什么这是一个好时机重构代码? 1. **信任度提升:** 如果您有足够...
总结起来,优雅的C#代码需要遵循良好的命名规范,控制方法参数数量,合理使用注释,避免数值硬编码,巧用`string.Empty`,以及适时重构。这些实践不仅提升代码质量,还增强团队间的沟通效率,从而提高整体项目开发的...
这篇文档集包含了电子科技大学唐续老师的SOPC实验代码与总结,为学习者提供了宝贵的实践经验和理论指导。 SOPC技术的核心在于FPGA(Field-Programmable Gate Array),这是一种可重构的数字逻辑器件,可以按照设计...
本篇将深入探讨OMP算法在压缩感知重构中的原理、实现及应用。 一、压缩感知理论基础 压缩感知理论基于两个关键假设:稀疏性和可分解性。稀疏性是指大多数实际信号都可以用较少的非零元素表示,即在某种基下,信号是...
- **可视化展示**:在MATLAB界面上,用户可以看到重构后的网络图。其中,红色线条表示被断开的支路,实线代表原有的33个节点间的线路,而虚线则代表了新增的联络线路。 #### 五、案例分析 - **IEEE33节点配电网**...
这篇总结包含了这些核心知识点,通过近千行的代码实例帮助初学者深入理解C语言。以下是针对内容的详细解释: 1. **字符数组与字符串定义**: - 字符数组`arr1`用于存储整型数值,使用`printf`进行格式化输出。 - ...
总结来说,这篇论文提供了一种新的视角来处理软件重构问题,特别是从设计层次出发,通过自动化工具和算法实现对软件结构的优化,旨在提高软件的可维护性、可读性和整体性能。这种方法为未来的软件开发和维护提供了有...
总结来说,这篇研究工作不仅在理论上提出了创新的编程范式,而且通过实践证明了其有效性。简化编程的设计范式有助于减轻程序员的负担,提高软件开发效率,并减少维护成本。在高性能计算和复杂数据处理领域,这种改进...
例如,使用代码自动完成功能可以快速输入代码,重构工具可以帮助我们进行代码重构,而调试工具则可以方便地定位和解决问题。学习并熟练掌握IDE的快捷键和定制化设置,能极大地提高开发效率。 在团队协作中,版本...
标题中的“我的年终项目总结(2篇)”表明这是一份关于作者在过去一年中在IT领域项目经验的总结,可能涵盖了技术应用、项目管理、团队协作等多个方面。结合描述中的博文链接,我们可以推测作者可能在博客中分享了他...
总结起来,这篇以“CS_OMP_博士论文_论文matlab_matlab_重构算法”为主题的压缩包内容,提供了一篇关于压缩感知重构算法——正交匹配追踪算法的matlab仿真代码,旨在帮助读者深入理解OMP算法的原理和实现过程,是...
这本书被视为《代码大全》的姊妹篇,承载了作者在微软公司超过十年的工作经验,是微软软件工程师必读之作。它深入探讨了软件开发流程、技术、方法论等多个层面的知识,旨在为软件开发者提供宝贵的经验教训。 首先,...
- **重构指导**:在进行代码重构时,统计工具可以帮助开发者识别哪些部分的代码可能需要重点关注。 4. **注意事项** - **代码质量不完全依赖于行数**:虽然代码行数是评估的一个维度,但代码质量更重要的是结构...
10. **代码重构与设计模式**:遵循面向对象设计原则,运用设计模式(如单例、工厂、观察者等),并进行定期的代码重构,可以保持代码的清晰性和可维护性。 通过阅读“VC编程经验总结.chm”文件,你可以深入学习和...
本篇文章将详细解析一个名为“jQuery投票征名代码”的项目,该项目利用jQuery 1.4.2版本实现了一个鼠标悬停时动态展示征名释义的效果。这一功能在互动性强的投票系统或在线活动中,能够提供用户友好的体验,增强用户...
本篇文章将详细探讨代码统计工具的功能、重要性以及如何使用SourceCounter这样的工具进行代码统计。 首先,代码统计工具的主要功能包括: 1. 计算总代码行数:这涵盖了源代码文件中的所有可执行语句,包括空格、...
#### 四、代码重构技巧 **原文摘要:** - **重构的重要性**:重构是提高代码质量和性能的有效手段之一。 - **重构策略**:在重构过程中,应保持原有功能不变的同时优化代码结构;重构之前应对原有代码进行全面测试...
5. **代码重构**:定期审查并重构代码,去除冗余和不必要的逻辑。 四、利用`NSObject`增强`UIViewController` `UIViewController`继承自`NSObject`,因此可以利用`NSObject`的一些特性来优化代码,如KVO(Key-Value...