IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> Java代码重构篇_02 重复代码 -> 正文阅读

[Java知识库]Java代码重构篇_02 重复代码


02重复代码

什么是重复代码?

定义:在一个以上的地点看到相同或相似的代码结构。
影响:想要修改其中一段代码逻辑需要修改多次,易遗漏,难维护
改进目标:消除重复,提升可维护性的目标。
方法:提炼函数,移动语句,函数上移等重构方法。

案例1:同一个类的两个函数含有相同的表达式

/**
 * 计算水果总价(同一个类的两个函数含有相同的表达式)
 *
 * @since 2021-08-18
 */
public class FruitsCost {
    public double computeMoneyWithoutPrivileges(String type, int numbers) {
        double prices;
        switch (type) {
            case "apple":
                prices = 5.5;
                break;
            case "banana":
                prices = 4.0;
                break;
            case "strawberry":
                prices = 10.5;
                break;
            default:
                throw new IllegalArgumentException("Illegal type : " + type);
        }
        return prices * numbers;
    }

    public double computeMoneyWithPrivileges(String type, double numbers, double discount) {
        double prices;
        switch (type) {
            case "apple":
                prices = 5.5;
                break;
            case "banana":
                prices = 4.0;
                break;
            case "strawberry":
                prices = 10.5;
                break;
            default:
                throw new IllegalArgumentException("Illegal type : " + type);
        }
        return prices * numbers * discount;
    }
}

代码背景:两个函数根据水果类型获得单价并计算水果费用;其中一个函数全价购买,一个函数打折购买;

症状/问题:两个函数在获得水果单价的代码是一致的,如果其中一
种水果单价出现变化,则需要同时修改两处

重构方法
我们重构的方法是这样,把相同部分的代码提取出来成一个方法,然后原代码处调用这个提取代码的方法即可。

如果你用的是idea,可以这样做:
选中重复的代码,按ctrl+shift+M(提取方法),出现一个窗口,我们可以修改访问级别,一般用private,毕竟是在类的内部使用,可以给方法命名,下面还有预览方法,确定好最后点refactor
在这里插入图片描述
然后他会提示其它方法的相同的代码(绿色部分),点replaces一起提取即可。
在这里插入图片描述

重构后

public class FruitsCost {
    public double computeMoneyWithoutPrivileges(String type, int numbers){
        return getPrices(type) * numbers;
    }

    public double computeMoneyWithPrivileges(String type, double numbers, double discount){
        return getPrices(type) * numbers * discount;
    }

    private double getPrices(String type) {
        double prices;
        switch (type) {
            case "apple":
                prices = 5.5;
                break;
            case "banana":
                prices = 4.0;
                break;
            case "strawberry":
                prices = 10.5;
                break;
            default:
                throw new IllegalArgumentException("Illegal type : " + type);
        }
        return prices;
    }
}

案例2:两个互为兄弟的子类含有相同的表达式

/**
 * 水果利润(两个互为兄弟的子类含有相同的表达式)
 *
 * @since 2021-08-18
 */
class Fruits {
    // 成本单价
    public double costPrices;

    // 出售单价
    public double prices;

    // 最小出货量
    public double minSaleableNum;
}
/**
 * 苹果利润(两个互为兄弟的子类含有相同的表达式)
 *
 * @since 2021-08-18
 */
class Apple extends Fruits {
    public Apple(double costPrices, double prices, double minSaleableNum) {
        this.costPrices = costPrices;
        this.minSaleableNum = minSaleableNum;
        this.prices = prices;
    }

    public double profitMoney(int number) {
        return Math.max(0, number - minSaleableNum) * this.prices - this.costPrices * number;
    }
}
/**
 * 香蕉利润(两个互为兄弟的子类含有相同的表达式)
 *
 * @since 2021-08-18
 */
class Banana extends Fruits {
    public Banana(double costPrices, double prices, double minSaleableNum) {
        this.costPrices = costPrices;
        this.minSaleableNum = minSaleableNum;
        this.prices = prices;
    }

    public double profitMoney(int number) {
        return Math.max(0, number - minSaleableNum) * this.prices - this.costPrices * number;
    }
}

代码背景:代码由水果父类,以及苹果香蕉两个子类组成。计算每种水果的销售利润。考虑运费、存储、破损等成本,计算毛利润时,需要从销量中减去最小出货量。

症状:两个子类计算利润的方法完全一致,如果计算利润方式出现变化,则需要修改多个子类。

重构方法
我们可以把这些子类都有的方法提取,放到父类中。比如这个例子,把所有水果类中的profitMoney方法都提取到Fruits类中,只写一次。当我们添加新的水果子类时,不需要再写该方法;当计算利润方式发生变化时,我们只需要修改Fruits类中的方法即可,只需要修改一次;当其中某中水果有特殊的计算方式,我们只需要重写该子类的方法。这正好适应了面向对象继承和方法重写的特性。

如果你用idea,你可以用快捷键来重构。
将鼠标光标放到需要提取的方法上,按ctrl+shift+alt+T,或者右键到refator中,找到Pull Members Up,点击Refoctor,这样代码就提取到父类中了。
在这里插入图片描述

重构后代码
把子类中的重复方法提取到父类中

class Fruits {
    // 成本单价
    public double costPrices;

    // 出售单价
    public double prices;

    // 最小出货量
    public double minSaleableNum;

    public double profitMoney(int number) {
        return Math.max(0, number - minSaleableNum) * this.prices - this.costPrices * number;
    }
}

案例3:代码片段中语句有可能增删改,功能不变

/**
 * 月份判断(代码片段中语句有可能增删改,功能不变)
 *
 * @since 2021-08-18
 */
class MonthJudgement {
    public boolean judgeMonth() {
        Long timeStamp = System.currentTimeMillis();  // 获取当前时间戳
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        String date = sdf.format(new Date(Long.parseLong(String.valueOf(timeStamp))));
        String month = date.split(" ")[0].split("-")[1];
        return "12".equals(month);
    }
}
/**
 * 年份判断(代码片段中语句有可能增删改,功能不变)
 *
 * @since 2021-08-18
 */
class YearJudgement {
    public boolean judgeYear() {
        Long time = System.currentTimeMillis();  // 获取当前时间戳
        System.out.println("获得当前时间戳");
        SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        String date = dateFormat.format(new Date(Long.parseLong(String.valueOf(time))));
        return date.startsWith("2021");
    }
}

代码背景
重复代码出现在不同的类中。代码逻辑不完全一致,变量名不同分别为time和timeState, judgeYear多 了一条打印语句。

症状
两个类中函数judgeMonth和judgeYear函数逻辑基本上是一致,只是个别不相同的地方,如果修改时间格式,需要同时修改两处。

重构方法
把鼠标放在代码上,按ctrl+shift+上/下,可以移动语句,在方法上可以调动位置。注意你需要先确认这样不会改变业务逻辑,才可进行。然后才可以提取相同的代码。还可以搬移函数,放到其他的类中,如果使用快捷键(F6),需要先将抽取出来的方法改为静态。
重构后代码

/**
 * 日期转换
 *
 * @since 2021-08-18
 */
public class DateFormatter {
    public static String getDate() {
        Long timeStamp = System.currentTimeMillis();  // 获取当前时间戳
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        return sdf.format(new Date(Long.parseLong(String.valueOf(timeStamp))));
    }
}
/**
 * 月份判断(代码片段中语句有可能增删改,功能不变)
 *
 * @since 2021-08-18
 */
class MonthJudgement {
    public boolean judgeMonth() {
        String date = DateFormatter.getDate();
        String month = date.split(" ")[0].split("-")[1];
        return "12".equals(month);
    }
}
/**
 * 年份判断(代码片段中语句有可能增删改,功能不变)
 *
 * @since 2021-08-18
 */
class YearJudgement {
    public boolean judgeYear() {
        System.out.println("获得当前时间戳");
        String date = DateFormatter.getDate();
        return date.startsWith("2021");
    }
}

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-05-01 15:33:33  更:2022-05-01 15:36:55 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 2:15:14-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码