首页 > 其他分享 >设计模式---策略模式

设计模式---策略模式

时间:2022-11-02 22:25:39浏览次数:46  
标签:target 模式 --- String input new 设计模式 public 客户端

简述

预先定义有着不同执行过程但结果相同算法族,运行时指定所需算法。

算法族
此处为一组有共同主题的有相同结果的不同算法的集合。

话不多说,看个优化案例。

优化案例

最初版v0

不使用策略模式的案例。四种不同的计算策略。客户端的代码如下。

// 客户端
public class Client {
    public static void main(String[] args) {
        String target = "公园";
        Scanner sc = new Scanner(System.in);
        String input = sc.next();
        if ("foot".equals(input)) {
            System.out.println("徒步到目的地:" + target);
        } else if ("bike".equals(input)) {
            System.out.println("骑自行车到目的地:" + target);
        } else if ("car".equals(input)) {
            System.out.println("开车到目的地:" + target);
        }
        sc.close();
    }
}

具体的条件分支都写在客户端,日后增加新的条件分支时也需要修改客户端。修改客户端这件事往往是不太愿意接受的。原因是我们希望客户端尽可能少的改变,以便减少客户使用系统的学习成本。

传统的方法就只能增加if条件判断了,如下。

修改版v1

只需修改客户端,其他代码不变。

// 客户端
public class Client {
    public static void main(String[] args) {
        String target = "公园";
        Scanner sc = new Scanner(System.in);
        String input = sc.next();
        if ("foot".equals(input)) {
            Foot foot = new Foot();
            foot.toTarget(target);
        } else if ("bike".equals(input)) {
            Bike bike = new Bike();
            bike.toTarget(target);
        } else if ("car".equals(input)) {
            Car car = new Car();
            car.toTarget(target);
        }
        sc.close();
    }
}

可以看出客户端依旧与各个具体的类耦合(从类的创建到方法的调用都是如此)。

可以使用策略模式优化,使得方法调用不需要if条件判断,传入什么样的对象就使用什么对象的行为。

修改版v2(策略模式)

public interface Trans {
    void toTarget(String target);
}

// 徒步去目的地
public class Foot implements Trans {
    @Override
    public void toTarget(String target) {
        System.out.println("徒步到目的地:" + target);
    }
}

// 骑自行车去目的地
public class Bike implements Trans {
    @Override
    public void toTarget(String target) {
        System.out.println("骑自行车到目的地:" + target);
    }
}

// 开车去目的地
public class Car implements Trans {
    @Override
    public void toTarget(String target) {
        System.out.println("开车到目的地:" + target);
    }
}

// 上下文类,根据客户端业务的需求持有不同的计算对象
public class Context {
    private Trans trans;
    
    public Context(Trans trans) {
       	this.trans = trans;
    }
    
    // 更改持有的计算对象
    public change(Trans trans) {
        this.trans = trans;
    }
    
    // 实际调用持有的trans实现计算
    public int toTarget(String target) {
        return trans.toTarget(target);
    }
}

修改后,客户端代码调用。

// 客户端
public class Client {
    public static void main(String[] args) {
        String target = "公园";
        Scanner sc = new Scanner(System.in);
        String input = sc.next();
        Context context = null;
        if ("foot".equals(input)) {
            context = new Context(new Foot());
        } else if ("bike".equals(input)) {
            context = new Context(new Bike());
        } else if ("car".equals(input)) {
            context = new Context(new Car());
        }
        System.out.println(context.toTarget(target));
        sc.close();
    }
}

代码量确实有一定的减少,但是客户端代码从只与各个具体Trans类的实现类耦合到多耦合一个上下文类,这样想与我们的需求背道而驰啊。实际上,单纯的策略模式就是如此,只负责减少方法调用的if语句,而不设计对象创建的封装与优化。

说到对象创建的优化,就得说到工厂模式了,事实上在使用策略模式时,为了创建对象也变得方便,通常也会使用到工厂模式进行优化。详情看以下优化案例。

修改版v3(策略+简单工厂)

现有代码都不需要改变,只需要使用简单工厂封装上下文对象的创建即可。

// 工厂类,创建持有不同Trans对象的上下文对象
public class Factory {
    public static Context create(String input) {
        if ("foot".equals(input)) {
            return new Context(new Foot());
        } else if ("bike".equals(input)) {
            return new Context(new Bike());
        } else if ("car".equals(input)) {
            return new Context(new Car());
        }
        return null;
    }
}

修改后,客户端代码调用。

// 客户端
public class Client {
    public static void main(String[] args) {
        String target = "公园";
        Scanner sc = new Scanner(System.in);
        Context context = Factory.create(sc.next());
        System.out.println(context.toTarget(target));
        sc.close();
    }
}

客户端代码大幅减少,并且客户端中仅仅与Context类存在耦合。创建与使用的核心逻辑都从客户端剥离,且具体调用的方法也只有在运行时才知晓(核心目的)。这样就能少些很多if语句了。

总结

优点

  • 可以大幅减少if语句的书写。
  • 增加新的实现方法也不需要修改客户端代码,只需要增加实现类。

缺点

  • 单纯的策略模式需要客户端对于各个实现类有足够的了解,提升了开发时对系统的理解难度。

  • 策略过多时,存在策略膨胀的问题。鉴于策略膨胀问题,应该慎用策略模式。这是使用混合模式或许可以解决这个问题。

    混合模式
    即在策略模式的实现类的方法中使用if语句分割各个情况。

适用场景

  • 想要优化系统中过多的if语句时。

标签:target,模式,---,String,input,new,设计模式,public,客户端
From: https://www.cnblogs.com/spoonb/p/16750785.html

相关文章

  • leetcode-104. 二叉树的最大深度
    题目描述给定一个二叉树,找出其最大深度。二叉树的深度为根节点到最远叶子节点的最长路径上的节点数。说明: 叶子节点是指没有子节点的节点。示例:给定二叉树[3,9,......
  • springboot项目整合-注册功能模块开发
    工程简介准备工作:项目所用到的html界面以及sql文件链接如下:链接:https://pan.baidu.com/s/18loHJiKRC6FI6XkoANMSJg?pwd=nkz2提取码:nkz2复制这段内容后打开百度网盘......
  • 【算法与数据结构2】数据结构基础----数组、列表
    一、物理结构数组  数组是存储相同数据类型的元素的一种有序数据结构,通过下标进行存储。查找的时间复杂度为O(1),而删除和添加的时间复杂度为O(n)。其代码实现如下:pu......
  • MYSQL-安装
    1、下载地址https://downloads.mysql.com/archives/community/2、解压3、下图目录下创建一个my.ini文件写入下方内容[mysql]default-character-set=utf8[mysqld]......
  • 论文笔记 - RETRIEVE: Coreset Selection for Efficient and Robust Semi-Supervised
    Motivation虽然半监督学习减少了大量数据标注的成本,但是对计算资源的要求依然很高(无论是在训练中还是超参搜索过程中),因此提出想法:由于计算量主要集中在大量未标注的数据上......
  • Week5-Technology: Internets and Packets
    Week5-Technology:InternetsandPacketsCommonLinkLayertechnologiesare…WiFiOpticalSatelliteEthernetCableModemDSL**Whenlookingataddressesf......
  • 2022-11-2学习内容
    1.外部存储空间1.1FileWriteActivity.javapackagecom.example.chapter06;importandroidx.appcompat.app.AppCompatActivity;importandroid.os.Bundle;importan......
  • javaSE基础-注解与枚举类
    注解与枚举类注解1、jdk5.0新增的功能2、Annotation就是代码的特殊标记,这些标记可以在编译、类加载、运行时被读取,并执行相应处理,通过使用注解程序员在可以不改变原有......
  • js 四级联动,利用的vue 的v-model双向绑定
    数据格式为:  <!DOCTYPEhtml><htmllang="en"><head><metacharset="UTF-8"><metaname="viewport"content="width=device-width,initial-scale=1.0......
  • Androidstudio快速解决Gradlesdependencycachemaybecorrupt和Gradle配置gradle-3.*-al
    Error:Failedtoopenzipfile. Gradle'sdependencycachemaybecorrupt(thissometimesoccursafteranetworkconnectiontimeout.) Re-downloaddependencies......