简单通俗java设计模式(二)策略模式

一、概念

策略(Strategy)模式的定义:该模式定义了一系列算法,并将每个算法封装起来,使它们可以相互替换,且算法的变化不会影响使用算法的客户。策略模式属于对象行为模式,它通过对算法进行封装,把使用算法的责任和算法的实现分割开来,并委派给不同的对象对这些算法进行管理。

二、学习准备

要先学习策略模式,首先要去了解2个接口ComparatorComparable
(1)Comparator:
通过jdk1.8的api文档了解:
Comparator的抽象方法

在这里插入图片描述
详细方法信息:

在这里插入图片描述

在这里插入图片描述
(2)Comparable
在这里插入图片描述

三、代码实现

通过一系列例子了解什么是策略思想
先写一个排序:

public class Main {
    public static void main(String[] args) {
        int [] a={9,2,3,5,7,1,4,6,8};
        Sorter sorter=new Sorter();
        sorter.sort(a);
        System.out.println(Arrays.toString(a));
    }
}

这里我使用的选择排序法:

public class Sorter {
    public  int[] sort(int[] arr) {
        if (arr == null && arr.length == 0) {
            return arr;
        }
        for (int i = 0; i < arr.length - 1; i++) {
            int min = i;
            for (int j = i + 1; j < arr.length; j++) {
                min = arr[min] > arr[j] ? j : min;
            }
            sweap(arr,i, min);
        }
        return arr;
    }

    public static void sweap(int[] arr, int i, int j) {
        int tmp = arr[i];
        arr[i] = arr[j];
        arr[j] = tmp;
    }
}

代码简单的实现了对int类型的数组排序,在问题来了,我要是对double类型进行排序,怎么办呢?
在写一个sort,把int类型改成double。如果是flot类型排序?又要重写一个sort方法。这样是不是很麻烦呢?
有不有一种更简单的写法呢?不用重写那么多sort?

继续看:

新建一个类Cat 实例:

public class Cat {
    @Override
    public String toString() {
        return "Cat{" +
                "weight=" + weight +
                ", height=" + height +
                '}';
    }

    int weight, height;

    public Cat(int weight, int height) {
        this.height = height;
        this.weight = weight;
    }

    public int compareTo(Cat c) {
        if (this.weight < c.weight) {
            return -1;
        } else if (this.weight > c.weight) {
            return 1;
        } else {
            return 0;
        }
    }
}

现在我们想,怎么在sort里面对Cat进行排序呢?
所以在Cat实例里面写一个 Cat的compareTo方法比较2个猫的weight,定义猫怎么比较大小。再将sort改成接收Cat类型的方法

public class Sorter {
    public void sort(Cat[] arr) {
        for (int i = 0; i < arr.length - 1; i++) {
            int min = i;
            for (int j = i + 1; j < arr.length; j++) {
                min = arr[j].compareTo(arr[min]) == -1 ? j : min;
            }
            sweap(arr, i, min);
        }
    }

    public static void sweap(Cat[] arr, int i, int j) {
        Cat tmp = arr[i];
        arr[i] = arr[j];
        arr[j] = tmp;
    }
}

看看结果:

public class Main {
    public static void main(String[] args) {
        Cat[] a={new Cat(3,3),new Cat(2,2),new Cat(4,4)};
        Sorter sorter=new Sorter();
        sorter.sort(a);
        System.out.println(Arrays.toString(a));
    }
}

在这里插入图片描述
现在我们对一个对象按照weight属性进行了排序,这时,我们要是对狗,对蛇 ,对人排序呢?岂不是又要修改sort方法吗?
所以在sort上写死类型是不行的。

就有人想到将sort接收类型改成Object不就行了吗?可Objerct里面没有compareTo方法呀。很简单
所以,我可以这样设计,所有要求实现sort方法的类型去重写里面的compareTo方法
直接去实现java.lang下面的Comparable接口,这个接口里面有个compareTo方法
详见头部api文档接口用法
自己手撸一个Comparable接口实现:
新建Comparable接口:

public interface Comparable {

    int compareTo(Object o);
}

新建一个Dog实例实现Comparable并重写compareTo方法:
注意:由于实现的compareTo方法为Object类型,则需要在重写的时候,需要强转为当前实例对象

public class Dog implements Comparable {
    int food;

    public Dog(int food) {
        this.food = food;
    }

    @Override
    public String toString() {
        return "Dog{" +
                "food=" + food +
                '}';
    }

    @Override
    public int compareTo(Object o) {
        Dog d= (Dog) o;
        if (this.food<d.food){
            return -1;
        }else if (this.food>d.food){
            return 1;
        }else {
            return 0;
        }
    }
}

在sort方法中将接收对象改为Comparable

public class Sorter {
    public void sort(Comparable[] arr) {
        for (int i = 0; i < arr.length - 1; i++) {
            int min = i;
            for (int j = i + 1; j < arr.length; j++) {
                min = arr[j].compareTo(arr[min]) == -1 ? j : min;
            }
            swap(arr, i, min);
        }
    }

    public static void swap(Comparable[] arr, int i, int j) {
        Comparable tmp = arr[i];
        arr[i] = arr[j];
        arr[j] = tmp;
    }
}

在看结果:

public class Main {
    public static void main(String[] args) {
        Dog[] a={new Dog(3),new Dog(5),new Dog(2)};
        Sorter sorter=new Sorter();
        sorter.sort(a);
        System.out.println(Arrays.toString(a));
    }
}

在这里插入图片描述
虽然我们用需要比较的实例重写了Comparable中的compareTo方法后,每次比较都不需要进行更改sort方法中的接收类型了

但是:发现没有?每次重写compareTo方法的时候,都需要强制转换。有时候还会出错。
所以我们可以采用java中泛型的写法:
修改Comparable接口:

public interface Comparable<T> {

    int compareTo(T o);
}

更改Dog实例:

public class Dog implements Comparable<Dog> {
    int food;

    public Dog(int food) {
        this.food = food;
    }

    @Override
    public String toString() {
        return "Dog{" +
                "food=" + food +
                '}';
    }

    @Override
    public int compareTo(Dog d) {
        if (this.food<d.food){
            return -1;
        }else if (this.food>d.food){
            return 1;
        }else {
            return 0;
        }
    }
}

在去执行main方法,结果就是我们想要的结果。
有了以上的基础后,我们再来了解什么是策略模式。

四、Comparator策略模式

虽然现在sort方法能去排序Comparable类型的对象,但是还不够灵活,在Cat中,我们通过weight来比较猫的大小,如果我们用hight来觉得它的大小呢?所以:我想猫比较大小的策略可以灵活的指定,这时候怎么做?

有人说,我把Cat重写的compareTo方法重新写不就行了吗?这相当于重新更改功能,如果每次都要更改,岂不是很麻烦?所以我们可以对compareTo方法进行拓展
Comparator:比较器,其实就是比较策略
新建Comparator接口

public interface Comparator<T> {
    
    int compare(T o1, T o2);
}

新建一个类的比较器:例如DogComparator

public class DogComparator implements Comparator<Dog> {
    @Override
    public int compare(Dog o1, Dog o2) {
        if (o1.food < o2.food) {
            return -1;
        } else if (o1.food > o2.food) {
            return 1;
        } else {
            return 0;
        }
    }
}

更改sort方法,接收类型为泛型,在加入比较器Comparator

public class Sorter<T> {
    public void sort(T[] arr, Comparator<T> comparator) {
        for (int i = 0; i < arr.length - 1; i++) {
            int min = i;
            for (int j = i + 1; j < arr.length; j++) {
                min = comparator.compare(arr[j], arr[min]) == -1 ? j : min;
            }
            swap(arr, i, min);
        }
    }

    public  void swap(T[] arr, int i, int j) {
        T tmp = arr[i];
        arr[i] = arr[j];
        arr[j] = tmp;
    }
}
public class Main {
    public static void main(String[] args) {
        Dog[] a={new Dog(3),new Dog(5),new Dog(2)};
        Sorter<Dog> sorter=new Sorter<>();
        sorter.sort(a,new DogComparator());
        System.out.println(Arrays.toString(a));
    }
}

通过实现Comparatorcompare方法,使代码拓展更强了
后面如果要拓展代码的比例方式,只需更改重写实例类compare就可以了。
举个例:
创建Cat的比较器CatWeightComparator
neight来比较猫的大小

public class CatWeightComparator implements Comparator<Cat> {
    @Override
    public int compare(Cat o1, Cat o2) {
        if (o1.neight< o2.neight) {
            return -1;
        } else if (o1.neight> o2.neight) {
            return 1;
        } else {
            return 0;
        }
    }
}
public class Main {
    public static void main(String[] args) {
        Cat[] a={new Cat(3,3),new Cat(2,2),new Cat(4,4)};
        Sorter<Cat> sorter=new Sorter<>();
        sorter.sort(a,new CatWeightComparator());
        System.out.println(Arrays.toString(a));
    }
}

我们可以通过实现Comparator接口比较器的策略,选择我们所需的策略进行排序
在这里插入图片描述
总结:策略模式封装的就是做一件事情不同的执行方式,比如排序的策略,根据不同的排序要求,选择不同的排序策略

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
尚硅谷是一家提供职业教育培训的机构,他们出版了《图解Java设计模式》这本书。 这本书主要介绍了Java设计模式的内容。设计模式是一种在软件设计中常用的解决问题的方法,它提供了一套通用的解决方案,帮助我们在开发过程中更加高效地解决各种问题。 《图解Java设计模式》从实际开发中的场景出发,通过图解的方式,为读者详细介绍了23种常用的设计模式。每一种设计模式都有对应的UML图和实例代码,从而帮助读者理解和应用这些设计模式。 这本书首先介绍了设计模式的基本概念和原则,然后详细介绍了创建型模式、结构型模式和行为型模式。在介绍每一种设计模式时,书中都会提供一个实际的应用场景,并通过图示和代码示例进行详细的讲解,帮助读者更好地理解和掌握这些设计模式。 通过阅读《图解Java设计模式》,读者可以学习到如何利用不同的设计模式解决实际开发中的问题,提高代码的复用性、可维护性和扩展性。这本书适合已经具备一定Java开发经验的读者,对于希望提升自己的设计能力和编码水平的开发者来说,是一本值得推荐的书籍。 总之,《图解Java设计模式》是一本内容丰富、通俗易懂、实用性强的Java设计模式入门书籍,通过阅读并实践书中的内容,读者可以提升自己的软件设计和开发能力。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值