`
381895649
  • 浏览: 230342 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

java 代码重构-第一章(分解并重组statement())

    博客分类:
  • java
 
阅读更多

 

上一篇文章:java 代码重构-第一章(起点)

下一篇文章:java 代码重构-第一章(类自己该做自己的事)

 

2.1 把switch代码在方法中抽出

  • 第一个明显引起我注意的就是长得离谱的statement() 。每当看到这样长长的函数,我就想把它大卸八块。要知道,代码区块愈小,代码的功能就愈容易管理,代码的处理和搬移也都愈轻松。

  • 重构过程的第一阶段中,我将说明如何把长长的函数切开,并把较小块的代码移至更合适的class 内。我希望降低代码重复量,从而使新的(打印HTML 报表用的)函数更容易撰写。

  • 第一个步骤是找出代码的逻辑泥团(logical clump)并运用 Extract Method。本例一个明显的逻辑泥团就是switch 语句,把它提炼(extract)到独立函数中似乎比较好。

  • 和任何重构准则一样,当我提炼一个函数时,我必须知道可能出什么错。如果我提炼得不好,就可能给程序引入臭虫。所以重构之前我需要先想出安全作法。

  • 首先我得在这段代码里头找出函数内的局部变量(local variables)和参数(parameters)。我找到了两个:each 和thisAmount,前者并未被修改,后者会被修改。任何不会被修改的变量都可以被我当成参数传入新的函数,至于会被修改的变量就需格外小心。如果只有一个变量会被修改,我可以把它当作返回值。thisAmount 是个临时变量,其值在每次循环起始处被设为0,并且在switch 语句之前不会改变,所以我可以直接把新函数的返回值赋予它。

  • 下面展示重构前后的代码。重构前的代码在下面类对比.凡是从函数提炼出来的代码,以及新代码所做的任何修改,只要我觉得不是明显到可以一眼看出,就以粗体字标示出来特别提醒你。

 

上一篇文章,最开始的代码

Customer 

 

/**
 * 通计清单
 * @return
 */
public String statement(){
    double totalAmount = 0;//合计
    int frequentRentePoints = 0;
    Enumeration<Rental> enu_rentals = rentals.elements();
    String result = "Rental Record for "+ this.getName() + " 
";
    while(enu_rentals.hasMoreElements()){
        double thisAmount = 0;
        Rental each = enu_rentals.nextElement();

        switch (each.getMovie().getPriceCode()) {
        case Movie.REGULAR:
            thisAmount += 2;
            if(each.getDeysRented() > 2){
                thisAmount += (each.getDeysRented() - 2) * 1.5;
            }
            break;
        case Movie.NEW_RELEASE:
            thisAmount += each.getDeysRented() * 3;
            break;
        case Movie.CHILDRENS:
            thisAmount += 1.5;
            if(each.getDeysRented() > 3){
                thisAmount += (each.getDeysRented() -3 ) * 1.5;
            }
            break;
        }

        frequentRentePoints ++;

        if((each.getMovie().getPriceCode() == Movie.NEW_RELEASE) && each.getDeysRented() > 1){
            frequentRentePoints ++;
        }

        result += "	"+each.getMovie().getTitle() + "	" +String.valueOf(thisAmount) + "
";

        totalAmount += thisAmount;

    }
    result += "Amount owed is " + String.valueOf(totalAmount) + "
";
    result += "You earned " + String.valueOf(frequentRentePoints) + " frequent renter points";
    return result;
}

 

 

跟下面重构的代码对比(Customer 完整代码)

 

package com.mkfree.refactoring.shap1;

import java.util.Enumeration;
import java.util.Vector;

/**
 * 顾客
 * @author hk
 *
 * 2012-12-25 下午10:59:03
 */
public class Customer {

    private String name;
    private Vector<Rental> rentals = new Vector<>();

    public Customer(String name) {
        this.name = name;
    }

    /**
     * 添加
     * @param rental
     */
    public void addRentals(Rental rental){
        rentals.add(rental);
    }

    public String getName() {
        return name;
    }

    /**
     * 通计清单
     * @return
     */
    public String statement(){
        double totalAmount = 0;//合计
        int frequentRentePoints = 0;
        Enumeration<Rental> enu_rentals = rentals.elements();
        String result = "Rental Record for "+ this.getName() + " 
";
        while(enu_rentals.hasMoreElements()){
            double thisAmount = 0;
            Rental each = enu_rentals.nextElement();

            thisAmount = amountFor(each);//计算一笔租片费

            frequentRentePoints ++;

            if((each.getMovie().getPriceCode() == Movie.NEW_RELEASE) && each.getDeysRented() > 1){
                frequentRentePoints ++;
            }

            result += "	"+each.getMovie().getTitle() + "	" +String.valueOf(thisAmount) + "
";

            totalAmount += thisAmount;

        }
        result += "Amount owed is " + String.valueOf(totalAmount) + "
";
        result += "You earned " + String.valueOf(frequentRentePoints) + " frequent renter points";
        return result;
    }

    /**
     * 把计算一笔租片
     * @param each
     * @return
     */
    private int amountFor(Rental each) { // 计算一笔租片费。
        int thisAmount = 0;
        switch (each.getMovie().getPriceCode()) {
        case Movie.REGULAR: // 普通片
            thisAmount += 2;
            if (each.getDeysRented() > 2)
                thisAmount += (each.getDeysRented() - 2) * 1.5;
            break;
        case Movie.NEW_RELEASE: // 新片
            thisAmount += each.getDeysRented() * 3;
            break;
        case Movie.CHILDRENS: // 儿童。
            thisAmount += 1.5;
            if (each.getDeysRented() > 3)
                thisAmount += (each.getDeysRented() - 3) * 1.5;
            break;
        }
        return thisAmount;
    }

}

 

 

 

每次做完这样的修改之后,我都要编译并测试。这一次起头不算太好——测试失败了,有两笔测试数据告诉我发生错误。一阵迷惑之后我明白了自己犯的错误。我愚蠢地将amountFor() 的返回值型别声明为int,而不是double 。

以上代码就已经消除了switch里的代码...是不是,整个类好看些呢呢?

 

 

本文章来自:http://blog.mkfree.com/posts/18

 

分享到:
评论

相关推荐

    java代码重构一到六章

    ### Java代码重构:掌握优化现有代码的艺术 #### 引言 在软件开发的过程中,随着项目的不断演进,代码库往往会出现复杂度增加、可读性和可维护性下降的问题。这时,进行代码重构变得至关重要。重构是指在不改变...

    重构-改善既有代码的设计

    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...

    .java代码重构

    代码重构是软件开发过程中的一个重要环节,它涉及到对现有代码的改进,以提高代码的可读性、可维护性,同时并不改变其外在行为。在Java编程中,代码重构是一种常见的实践,尤其在大型项目中,为了保持代码的健康状态...

    java代码重构经验分享

    Java 代码重构经验分享 Java 代码重构是指在不改变外部行为的情况下,修改代码的内部结构,以提高代码的可维护性、可读性和可扩展性。本文总结了 Java 代码重构的经验和技术规范,包括重构要求、重构的工作、代码的...

    重构-第3章 代码的坏味道-读书笔记

    《重构-第3章 代码的坏味道》是软件开发领域的一本经典著作,由Martin Fowler所著。这本书深入探讨了如何识别并消除代码中的不良设计模式,以提高代码质量、可读性和可维护性。在第三章中,作者详细列举了多种"代码...

    java代码重构经验总结

    本文将深入探讨Java代码重构的关键点,涵盖重构原则、重构技巧以及常见的代码“坏味道”(bad smells),并结合示例代码进行分析。 #### 一、重构的重要性与原则 **1. 重构时机** - **版本控制前的重构:** 在...

    一本介绍如何将现有的 Java 代码重构为 Kotlin 代码的书籍

    本书《从Java到Kotlin:重构指南》由Duncan McGregor与Nat Pryce共同编写,是一本专为已经熟悉Java编程语言并希望将其现有代码转换为Kotlin的开发者准备的书籍。书中不仅提供了丰富的重构技巧,还包括了实用的示例和...

    重构--改善既有代码的设计_中文版

    java开发经典书籍,重构--改善既有代码的设计_中文版 java开发经典书籍,重构--改善既有代码的设计_中文版 java开发经典书籍,重构--改善既有代码的设计_中文版

    Java 代码重构示例

    Java代码重构示例 Java代码重构示例 Java代码重构示例 Java代码重构示例 Java代码重构示例 Java代码重构示例 Java代码重构示例

    重构-改善既有代码的设计.pdf

    重构-改善既有代码的设计.pdf重构-改善既有代码的设计.pdf

    Java 代码重构实例

    Java代码重构是一种优化编程实践,旨在改进代码的结构和可读性,而不改变其外部行为。重构对于提高软件质量和维护性至关重要,尤其是在大型项目中。以下是一些在Java重构中的关键原则和技巧,通过实例来展示如何应用...

    java代码重构圣经

    java代码重构以前忽视了,最近在看 字字珠玑,相见恨晚

    《重构——改善既有代码的设计》第一个案例代码

    《重构——改善既有代码的设计》是Martin Fowler的经典著作,它为软件开发人员提供了一套系统化的重构技术,旨在提升代码质量、可读性和维护性。在这个案例中,我们聚焦于一个影片出租点的程序,通过逐步的重构过程...

    代码重构-支持业务发展

    性能提升: 为了提升性能重构代码 业务发展: 业务发展,需要对现有项目进行拆分或者合并 维护困难: 代码冗余,难以读懂,性能低下,陷阱很多 框架设计瓶颈: 起初的设计方案和框架遇到瓶颈

    代码重构-重构是设计演进的基本方法

    软件设计是演进过程,而重构是设计演进的基本方法。重构是指不改变软件行为的前提下,修改程序内部结构。重构说简单,做不简单。首先,需要知道代码的好坏,即代码异味,设计原则等。其次,需要以自动测试作为保障。

    java代码重构

    Java代码重构是一个重要的软件开发实践,它涉及到对现有代码的改进,目的是提高代码的可读性、可维护性和整体质量,而不改变其外在行为。重构对于大型项目尤其关键,因为随着时间的推移,代码可能会变得复杂且难以...

    重读martin大师的《重构--提高既有代码的设计》第一章

    《重构--提高既有代码的...总之,Martin Fowler的《重构--提高既有代码的设计》第一章为我们提供了重构的基本理念和实践指南,通过遵循这些原则和技术,开发者可以逐步改善代码质量,实现更高效、更可持续的软件开发。

    重构-改善既有代码设计

    《重构-改善既有代码设计》是一本专注于软件开发过程中代码优化和重构的权威书籍,尤其对Android、Java、C和C++等编程语言的开发者具有很高的参考价值。本书旨在通过详细的实例和深入的分析,教导读者如何有效地改进...

Global site tag (gtag.js) - Google Analytics