`
jgsj
  • 浏览: 1051395 次
文章分类
社区版块
存档分类
最新评论

策略模式

 
阅读更多

  一、定义

  策略模式定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换。策略模式让算法独立于使用它的客户而独立变化。

  以上的定义会比较抽象。下面我来说一个具体的例子。假设有很多个人早上去上学,每个人都会有如何去学校的策略(如走路、公交、骑车),同学到了学校以后,老师把他们来学校的方法记录下来。假设有A、B、C三类学生,分别是走路、公交、骑车来学校,由于每一个学生类一定有来学校的策略,因此他们可以继承于同一个接口,接口中定义一个来学校的方法,接下来,只要老师能够调用这个方法,就能够知道他们各自来学校的方法了。

  C#代码如下:

using System;
namespace Demo
{
    interface IHowToGoToSchool
    {
        void HowToGo();
    }

    class StudentA : IHowToGoToSchool
    {
        public void HowToGo()
        {
            Console.WriteLine("By bus.");
        }
    }

    class StudentB : IHowToGoToSchool
    {
        public void HowToGo()
        {
            Console.WriteLine("By bike.");
        }
    }

    class StudentC : IHowToGoToSchool
    {
        public void HowToGo()
        {
            Console.WriteLine("Walking.");
        }
    }

    class Teacher
    {
        public void Record(IHowToGoToSchool method)
        {
            method.HowToGo();
        }
    }

    public class Program
    {
        public static void Main(string[] args)
        {
            Teacher teacher = new Teacher();
            StudentA a = new StudentA();
            StudentB b = new StudentB();
            StudentC c = new StudentC();
            teacher.Record(a);
            teacher.Record(b);
            teacher.Record(c);
        }
    }
}

  Java的代码如下:

interface IHowToGoToSchool
{
    void HowToGo();
}

class StudentA implements IHowToGoToSchool
{
    public void HowToGo()
    {
        System.out.println("By bus.");
    }
}

class StudentB implements IHowToGoToSchool
{
    public void HowToGo()
    {
    	System.out.println("By bike.");
    }
}

class StudentC implements IHowToGoToSchool
{
    public void HowToGo()
    {
    	System.out.println("Walking.");
    }
}

class Teacher
{
    public void Record(IHowToGoToSchool method)
    {
        method.HowToGo();
    }
}

public class Program
{
    public static void main(String[] args)
    {
        Teacher teacher = new Teacher();
        StudentA a = new StudentA();
        StudentB b = new StudentB();
        StudentC c = new StudentC();
        teacher.Record(a);
        teacher.Record(b);
        teacher.Record(c);
    }
}

  我们可以看到,使用策略模式的好处在于降低了程序的耦合度。Teacher类中的Record会传入一个继承了IHowToGoToSchool的对象,因此这个对象一定会有HowToGo这个方法。我们不必在Teacher类中关心这个方法的实现——因为它有StudentA、StudentB、StudentC各自的实现方法。如果我们需要增加一个新的学生类StudentD,只需要继承IHowToGoToSchool,然后实现HowToGo方法即可。

  二、另一个例子——方法回调(函数回调)

  设计模式中许多模式是类似的。下面举一个用策略模式来实现一个方法回调。(回调机制到底是单独算一个设计模式,还是算策略模式,还是算观察者模式,在我看来存在争议,主要是在于其用在什么场景下而定。就这个例子而言,我觉得它代表了策略模式的思想)

  C#代码如下:

using System;
namespace Demo
{
    public interface ICallBack
    {
        void Invoke();
    }

    public class Downloader
    {
        public static void Download(String url, ICallBack callback)
        {
            Console.WriteLine("正在下载:{0}", url);
            callback.Invoke();
        }
    }

    public class DownloadedShutdown : ICallBack
    {
        public void Invoke()
        {
            Console.WriteLine("下载完成,现在即将关机");
        }
    }

    public class DownloadedGetNext : ICallBack
    {
        public void Invoke()
        {
            Console.WriteLine("下载完成,现在下载下一个文件");
        }
    }

    public class Program
    {
        public static void Main(string[] args)
        {
            ICallBack shutDown = new DownloadedShutdown();
            ICallBack getNext = new DownloadedGetNext();
            Downloader.Download("www.qq.com/s.html", shutDown);
            Downloader.Download("www.qq.com/s.html", getNext);
        }
    }
}

  Java代码如下:

interface ICallBack
{
    void Invoke();
}

class Downloader
{
    public static void Download(String url, ICallBack callback)
    {
        System.out.println("正在下载:" + url);
        callback.Invoke();
    }
}

class DownloadedShutdown implements ICallBack
{
    public void Invoke()
    {
    	System.out.println("下载完成,现在即将关机");
    }
}

class DownloadedGetNext implements ICallBack
{
    public void Invoke()
    {
    	System.out.println("下载完成,现在下载下一个文件");
    }
}

public class Program
{
    public static void main(String[] args)
    {
        ICallBack shutDown = new DownloadedShutdown();
        ICallBack getNext = new DownloadedGetNext();
        Downloader.Download("www.qq.com/s.html", shutDown);
        Downloader.Download("www.qq.com/s.html", getNext);
    }
}


  我们在执行Downloader.Download时,传入了ICallback,在Download方法执行到最后,会调用这个ICallback对象的Invoke方法,因此我们可以看成,代码中的DownloadedShutdown类和DownloadedGetNext类都是回调的策略,它们分别表示了下载完后关机、下载完后继续下载两种策略。

  三、C#中策略模式的替代方法

  在C#中,你可能会想,这种回调完全可以用委托、事件,甚至Lambda表达式来实现,事实上确实这样能够简化很多代码。在C#中有代表意义的就是多线程类(Thread)。下面是一个C#多线程最简单的例子:

using System;
using System.Threading;
namespace Demo
{
    public class Program
    {
        public static void Main(string[] args)
        {
            Thread t = new Thread(ThreadMethod);
            t.Start();
        }

        public static void ThreadMethod()
        {
            Console.WriteLine("调用了一个多线程方法。");
        }
    }
}

  C#中存在方法的委托,因此可以把方法名当做参数传入,十分方便,而且我们不用为了实现某个接口而重新写一个类。

  另外一个例子是C#中的泛型容器List<T>.Sort方法。它是用于排序的一个方法,由于泛型容器的类型不确定,为了满足对特定类型对象进行排序,可以在Sort方法中传入比较器。比较器主要有两种,第一种比较器是一个继承了IComparer<T>接口的对象,这个就如之前所提到,我们要构造一个继承了这个接口的类,然后实现Compare方法,在调用List<T>.Sort的时候,它就会通过调用IComparer<T>.Compare来实现排序;第二种比较器是Comparison<in T>,它是一个委托,也就是说,你不用辛辛苦苦重新写一个新的类,而是只要写一个满足Comparison<in T>委托的方法就可以了。

  一下有个例子。每个学生都拥有一个Id,现在将学生簿List<Student>中的所有元素按照学生Id进行排序。下面将用C# 分别为Sort方法传入IComparer<T>和Comparison<in T>来实现排序:

using System.Collections.Generic;
using System;
namespace Demo
{
    public class Student
    {
        public int Id { get; set; }
        public Student(int id)
        {
            Id = id;
        }
    }

    public class StudentSortStrategy : IComparer<Student>
    {
        public int Compare(Student x, Student y){
            return x.Id - y.Id;
        }
    }

    public class Program
    {
        public static void Main(string[] args)
        {
            List<Student> students = new List<Student>();
            students.Add (new Student(5));
            students.Add (new Student(2));
            students.Add(new Student(3));
            Console.WriteLine ("Students after sorted:");
            students.Sort(new StudentSortStrategy());
            foreach (Student s in students)
            {
                Console.WriteLine(s.Id );
            }

            List<Student> students2 = new List<Student>();
            students2.Add(new Student(5));
            students2.Add(new Student(2));
            students2.Add(new Student(3));
            Console.WriteLine("Students2 after sorted:");
            students2.Sort(Compare);
            foreach (Student s in students2)
            {
                Console.WriteLine(s.Id);
            }
        }

        private static int Compare(Student x, Student y)
        {
            return x.Id - y.Id;
        }
    }
}


  输出结果为:

Students after sorted:

2

3

5

Students2 after sorted:

2

3

5

  可见,在C#中使用委托(方法名)可以实现策略模式。

  四、关于Java的策略模式

  Java中没有委托这么一说,因此策略模式就显得十分有用了。例如:在Java中实现多线程有两种方法,第一种是继承Thread类,第二种是继承Runnable接口。Runnable接口中定义了方法run,我们需要做的,是实现这个run,在调用Thread对象的start方法时,其实它是用多线程的方法调用了传入的Runnable接口对象的run方法。下面给出继承于Runnable接口的Java代码:

public class Program implements Runnable{
	public static void main(String[] args){
		Thread t = new Thread(new Program());
		t.start();
	}

	@Override
	public void run() {
		System.out.println("调用了一个多线程方法。");
	}
}

  在C#中,我们直接为Thread的构造函数中传入一个委托即可,而事实上,C#中的Thread类为密封类(也就是Java中的final class),我们无法进行继承,也无需进行继承。

  五、总结

  策略模式是一个很常用的模式,尤其是对于Java这种没有直接的委托机制的语言来说。设计模式中有许多模式的实现是相近的,因此在合适的场景下灵活地使用好它们是最重要的。

分享到:
评论

相关推荐

    策略模式结合模板方法模式

    策略模式结合模板方法模式的设计思路 策略模式结合模板方法模式是策略模式的一种变形,目的是为了解决策略模式中的一些共性问题。在策略模式中,经常会出现这样一种情况,就是发现这一系列算法的实现上存在公共功能...

    详解SpringBoot结合策略模式实战套路

    SpringBoot结合策略模式实战套路 策略模式是一种常用的设计模式,它可以使我们的代码更加灵活、可维护和可扩展。在SpringBoot项目中,策略模式可以与依赖注入机制相结合,实现更加灵活的业务逻辑处理。在本文中,...

    桥接模式和策略模式的区别,内含可运行代码和两者详细区别

    桥接模式和策略模式是软件设计模式中的两种重要模式,它们在实现上有着相似之处,但各自的应用场景和设计理念有所不同。下面将详细阐述这两种模式的特点、区别以及它们在实际编程中的应用。 首先,桥接模式(Bridge...

    设计模式之策略模式 鸭子问题

    设计模式之策略模式 鸭子问题 策略模式是一种经典的设计模式,通过鸭子问题,可以让学习者更好地了解设计模式的概念和实现。策略模式的主要思想是定义一系列的算法,并将每一个算法封装起来,使它们可以相互替换。...

    策略模式在实际项目中的应用二

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式通过定义一系列的算法,并将每一个算法封装起来,使它们可以相互替换,让算法独立于使用它的客户而变化。这种模式通常用于处理多种...

    Spring下使用策略模式

    在Spring框架中,策略模式是一种常见的设计模式,它允许我们定义一组可互换的策略,这些策略可以在运行时根据需求动态选择。这篇文章将深入探讨如何在Spring中运用策略模式,并结合源码分析其工作原理。 策略模式的...

    策略模式的实现,通过反射

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式通常用于将算法封装到不同的类中,使得可以根据需要动态选择并应用这些算法。本示例将详细介绍如何通过两种方法实现策略模式:一种...

    抽象工厂模式+工厂方法模式+策略模式+类图实现手机加工厂

    本文将探讨三个重要的设计模式:抽象工厂模式、工厂方法模式以及策略模式,并结合一个实际的场景——手机加工厂,来具体阐述它们的应用。 首先,我们来看**抽象工厂模式**。这个模式主要用于创建相关或依赖对象的...

    55-Java设计模式之策略模式与状态模式1

    Java 设计模式之策略模式与状态模式 策略模式是 Java 中的一种设计模式,它主要用于解决系统与第三方接口进行数据交互的问题。当系统需要与多种格式的数据进行交互时,使用策略模式可以很好地解决这个问题。例如,...

    策略模式封装的几个加密解密算法源码

    在"策略模式封装的几个加密解密算法源码"中,我们主要关注的是如何使用策略模式来封装常见的加密解密算法,如BASE64和MD5。 1. **BASE64编码**:BASE64是一种用于将二进制数据编码为ASCII字符的编码方式,以便在...

    策略模式的简单例子

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在软件开发中,我们经常遇到需要根据不同条件或选择执行不同算法的情况。策略模式提供了一种将算法封装到独立可互换的策略对象中,使得算法的变化独立...

    设计模式之策略模式,商场收银,封装算法

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在软件开发中,我们经常遇到需要根据不同的条件或场景来执行不同算法的情况。策略模式就是为了解决这类问题而提出的,它将每种算法封装到具有共同接口...

    js策略模式和代理模式

    策略模式和代理模式是设计模式中的两种常见模式,它们在软件开发中扮演着重要的角色,尤其是在JavaScript中,这两种模式提供了更加灵活和可维护的代码结构。 策略模式(Strategy Pattern)是一种行为设计模式,它...

    策略模式 template模式

    策略模式(Template模式) 策略模式是设计模式中的一种 객체行为型模式,它定义了一系列算法,封装每一个算法,并使它们可以互相替换。策略模式使得算法可以独立于使用它的客户而变化。 概述 在软件开发中,经常...

    Java 设计模式 策略模式

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式主要通过定义一系列的算法,并将每一个算法封装起来,使它们可以互相替换,让算法独立于使用它的客户而变化。 首先,策略模式的...

    策略模式的示例代码和思想模式

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式通常涉及接口或抽象类的实现,允许程序在运行时选择并应用不同的算法或策略。这种模式的核心在于将算法封装到独立的可互换的策略中...

    Java策略模式+案例

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式允许我们定义一组算法或策略,并将每个策略封装为一个类,使得它们可以互换,而不会影响到客户端代码。这种模式的核心在于"策略",...

    策略模式代码实现

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在策略模式中,一个类的行为或其算法可以在运行时更改。这种类型的设计模式属于行为模式。 策略模式定义了一系列的算法,并将每一个算法封装起来,使...

    软件设计模式策略模式实例

    策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在软件工程中,当一个系统需要在不同时间执行不同的算法或者行为时,策略模式就显得尤为有用。这种模式将算法封装到独立的可相互替换的策略类中,使得...

    设计模式解读之--策略模式.pdf

    ### 设计模式解读之策略模式 #### 模式定义与核心思想 策略模式是一种行为设计模式,它使得算法可以独立于使用它的客户而变化。在软件工程中,策略模式允许一组算法封装成一系列的类,它们共享相同的接口,但内部...

Global site tag (gtag.js) - Google Analytics