连载:面向对象葵花宝典:思想、技巧与实践(39) - 设计原则 vs 设计模式

简介:

又是设计原则,又是设计模式,到底该用哪个呢? =============================================================================

在“设计模型”一章中,我们提到设计原则和设计模式是互补的,设计原则和设计模式互补体现在:设计原则主要用于指导“类的定义”的设计,而设计模式主要用于指导“类的行为”的设计。

 

举一个很简单的例子:假设我们要设计一个图形类Shape,这个类既支持三角形,又支持矩形,其代码如下:

package com.oo.designpattern.diagram;

/**
 * 设计不好的Shape类,同时兼顾三角形和矩形的职责,不符合SRP设计原则
 *
 */
public class BadShape {

    //三角形的属性
    Position a;
    Position b;
    Position c;
    
    //矩形的属性
    Position m;
    int length;
    int width;
    
    public void drawTriangle(){
        //TODO: 画出三角形
    }
    
    public void drawRectangle(){
        //TODO: 画出矩形
    }
}


有经验的朋友都会觉得这个设计不太合理,因为其不符合类的SRP设计原则。因此,合理的做法是将这个类按照SRP原则拆分,具体拆分方法如下:

NormalShape.java

package com.oo.designpattern.diagram;

/**
 * 将BadShape拆开为三角形和矩形两个图形,并提取出NormalShape这个父类
 *
 */
abstract class NormalShape {
    abstract void draw();
}

NormalTriangle.java

package com.oo.designpattern.diagram;

/**
 * 三角形类
 *
 */
public class NormalTriangle extends NormalShape {
   
    //三角形的属性
    Position a;
    Position b;
    Position c;
    
    @Override
    public void draw() {
        // TODO:绘画三角形
        if(Config.CURRENT_SYSTEM == Config.WINDOWS){
            //TODO: 调用Windows的画图方法
        }
        else if( Config.CURRENT_SYSTEM == Config.LINUX){
            //TODO: 调用Linux的画图方法
        }
        else if( Config.CURRENT_SYSTEM == Config.MAC){
            //TODO: 调用Mac的画图方法
        }
    }

}

NormalRectangle.java

package com.oo.designpattern.diagram;

/**
 * 矩形类
 *
 */
public class NormalRectangle extends NormalShape {

    //矩形的属性
    Position m;
    int length;
    int width;
    
    @Override
    public void draw() {
        // TODO: 绘画矩形
        if(Config.CURRENT_SYSTEM == Config.WINDOWS){
            //TODO: 调用Windows的画图方法
        }
        else if( Config.CURRENT_SYSTEM == Config.LINUX){
            //TODO: 调用Linux的画图方法
        }
        else if( Config.CURRENT_SYSTEM == Config.MAC){
            //TODO: 调用Mac的画图方法
        }
    }

}


这样拆分之后,从类的设计原则来看,已经是符合要求了。

 

接下来我们再使用设计模式来继续完善这个设计,这里就需要使用设计模式之道来指导我们设计了,即:找到变化,封装变化

 

关于图形类一个比较明显的变化是跨平台,比如说要同时支持Windows、Linux、Mac三个桌面操作系统,那么实际画图的方法和需要调用的函数可能就随着平台的不同而变化,因此我们要找出一种方法来封装这种变化。

 

参考《设计模式》,可以知道这种方法就是“Bridge模式”,使用了Bridge后,会多出几个接口和实现类。

具体实现如下:

GoodShape.java

package com.oo.designpattern.diagram;

/**
 * 在NormalShape的基础上,增加Bridge设计模式的实现,使其更加适应于跨平台
 *
 */
abstract public class GoodShape {

    protected ShapeDraw _draw;  //将不同平台的实现封装到一个新的接口ShapeDraw
    abstract void draw();
}


GoodTriangle.java

package com.oo.designpattern.diagram;

/**
 * 按照Bridge设计模式设计的三角形类
 *
 */
public class GoodTriangle extends GoodShape {

    GoodTriangle(ShapeDraw draw){
        this._draw = draw;
    }
    
    @Override
    void draw() {
        // TODO Auto-generated method stub
        this._draw.drawTriangle();
    }

}

GoodRectangle.java

package com.oo.designpattern.diagram;

/**
 * 按照Bridge设计模式设计的矩形类
 *
 */
public class GoodRectangle extends GoodShape {

    GoodRectangle(ShapeDraw draw){
        this._draw = draw; 
    }
    
    @Override
    void draw() {
        // TODO Auto-generated method stub
        this._draw.drawRectangle();
    }

}

ShapeDraw.java

package com.oo.designpattern.diagram;

/**
 * 按照Bridge设计模式进行设计的画图的接口,封装了跨平台不同的实现
 *
 */
interface ShapeDraw {

    public void drawTriangle();
    public void drawRectangle();
}

WindowsDraw.java

package com.oo.designpattern.diagram;

/**
 * Windwos上的画图实现
 *
 */
public class WindowsDraw implements ShapeDraw {

    @Override
    public void drawTriangle() {
        // TODO Auto-generated method stub

    }

    @Override
    public void drawRectangle() {
        // TODO Auto-generated method stub

    }

}

LinuxDraw.java

package com.oo.designpattern.diagram;

/**
 * Linux上的画图实现
 *
 */
public class LinuxDraw implements ShapeDraw {

    @Override
    public void drawTriangle() {
        // TODO Auto-generated method stub

    }

    @Override
    public void drawRectangle() {
        // TODO Auto-generated method stub

    }

}

MacDraw.java

package com.oo.designpattern.diagram;

/**
 * Mac上的画图实现
 *
 */
public class MacDraw implements ShapeDraw {

    @Override
    public void drawTriangle() {
        // TODO Auto-generated method stub

    }

    @Override
    public void drawRectangle() {
        // TODO Auto-generated method stub

    }

}

可以看到,按照设计原则和设计模式进行重构后,原来不合理的设计逐步演变为一个优秀的设计了


================================================ 
转载请注明出处:http://blog.csdn.net/yunhua_lee/article/details/38655873
================================================ 

相关文章
|
3月前
|
设计模式 算法 搜索推荐
后端开发中的设计模式应用与实践
在软件开发的广袤天地中,后端技术如同构筑高楼大厦的钢筋水泥,支撑起整个应用程序的骨架。本文旨在通过深入浅出的方式,探讨后端开发领域内不可或缺的设计模式,这些模式犹如精雕细琢的工具箱,能够助力开发者打造出既健壮又灵活的系统架构。从单例模式到工厂模式,从观察者模式到策略模式,每一种设计模式都蕴含着深刻的哲理与实践价值,它们不仅仅是代码的组织方式,更是解决复杂问题的智慧结晶。
|
4月前
|
设计模式 算法 测试技术
PHP中的设计模式:策略模式的应用与实践
在软件开发的浩瀚海洋中,设计模式如同灯塔,指引着开发者们避开重复造轮子的暗礁,驶向高效、可维护的代码彼岸。今天,我们将聚焦于PHP领域中的一种重要设计模式——策略模式,探讨其原理、应用及最佳实践,揭示如何通过策略模式赋予PHP应用灵活多变的业务逻辑处理能力,让代码之美在策略的变换中熠熠生辉。
|
2月前
|
设计模式 算法 开发者
探索编程语言中的设计模式:从理论到实践
设计模式,这一编程世界中的灯塔,为无数开发者照亮了复杂问题解决的道路。本文将深入探讨设计模式在编程实践中的运用,以代码示例揭示其背后的智慧。无论你是初学者还是资深开发者,都能在这里找到启发和共鸣。让我们一起领略设计模式的魅力,开启编程世界的新篇章!
|
2月前
|
设计模式 监控 算法
Python编程中的设计模式应用与实践感悟###
在Python这片广阔的编程疆域中,设计模式如同导航的灯塔,指引着开发者穿越复杂性的迷雾,构建出既高效又易于维护的代码结构。本文基于个人实践经验,深入探讨了几种核心设计模式在Python项目中的应用策略与实现细节,旨在为读者揭示这些模式背后的思想如何转化为提升软件质量的实际力量。通过具体案例分析,展现了设计模式在解决实际问题中的独特魅力,鼓励开发者在日常编码中积极采纳并灵活运用这些宝贵的经验总结。 ###
|
2月前
|
设计模式 开发者 Python
Python编程中的设计模式应用与实践感悟####
本文作为一篇技术性文章,旨在深入探讨Python编程中设计模式的应用价值与实践心得。在快速迭代的软件开发领域,设计模式如同导航灯塔,指引开发者构建高效、可维护的软件架构。本文将通过具体案例,展现设计模式如何在实际项目中解决复杂问题,提升代码质量,并分享个人在实践过程中的体会与感悟。 ####
|
3月前
|
设计模式 API 持续交付
深入理解微服务架构:设计模式与实践
【10月更文挑战第19天】介绍了微服务架构的核心概念、设计模式及最佳实践。文章详细探讨了微服务的独立性、轻量级通信和业务能力,并介绍了聚合器、链式和发布/订阅等设计模式。同时,文章还分享了实施微服务的最佳实践,如定义清晰的服务边界、使用API网关和服务发现机制,以及面临的挑战和职业心得。
|
3月前
|
设计模式 存储 数据库连接
PHP中的设计模式:单例模式的深入解析与实践
在PHP开发中,设计模式是提高代码可维护性、扩展性和复用性的关键技术之一。本文将通过探讨单例模式,一种最常用的设计模式,来揭示其在PHP中的应用及优势。单例模式确保一个类仅有一个实例,并提供一个全局访问点。通过实际案例,我们将展示如何在PHP项目中有效实现单例模式,以及如何利用这一模式优化资源配置和管理。无论是PHP初学者还是经验丰富的开发者,都能从本文中获得有价值的见解和技巧,进而提升自己的编程实践。
|
3月前
|
设计模式 算法 PHP
PHP中的设计模式:策略模式的深入解析与实践
【10月更文挑战第9天】 策略模式是一种行为设计模式,它允许在运行时选择算法的行为。在PHP开发中,通过使用策略模式,我们可以轻松切换算法或逻辑处理方式而无需修改现有代码结构。本文将深入探讨策略模式的定义、结构以及如何在PHP中实现该模式,并通过实际案例展示其应用价值和优势。
39 1
|
3月前
|
设计模式 算法 PHP
PHP中的设计模式:策略模式的深入解析与实践
在PHP开发中,设计模式是提高代码可读性、可维护性和扩展性的重要工具。本文将深入探讨策略模式这一行为型设计模式,通过分析其定义、结构、使用场景以及在PHP中的实际应用,帮助开发者更好地理解和运用策略模式来优化自己的项目。不同于传统摘要的简洁概述,本文摘要部分将详细阐述策略模式的核心理念和在PHP中的实现方法,为读者提供清晰的指引。
|
3月前
|
设计模式 存储 测试技术
PHP中的设计模式:单例模式的深入解析与实践
在PHP开发领域,设计模式是解决常见问题的最佳实践。本文将深入探讨单例模式,一种确保类只有一个实例的设计模式,并提供实际应用示例。我们将从单例模式的基本概念讲起,通过实际案例展示如何在PHP中实现单例模式,以及它在不同场景下的应用和优势。最后,我们会探讨单例模式的优缺点,帮助开发者在实际项目中做出明智的选择。