策略模式简单谈

面向对象的编程不可避免的需要使用到各种各样的设计模式,设计模式可以使得代码更优雅,变得更加有弹性,设计模式带来的好处不需要再多说了,作者原先写的是Lua代码,函数式编程语言,没有很强的面向对象,因此很少会使用到设计模式。

最近开始接触Java的游戏服务端,发现里面的代码虽然看得懂,但是总是很难理解,不知道其为什么这样设计,后面慢慢发现里面运用到了很多设计模式,为了更好的理解代码,作者开始学习设计模式,在此记文。

首先接触到的是策略模式,我会用自己的理解来写一个简单的例子,就不说很多理论了。我发现大学时读过的设计模式,那个时候觉得应用性不足,因为看了觉得很有道理,但是很难联想其应用场景,老是会有疑问,这个到底用在哪里。但是当做过的项目多了之后,这些疑问就会得到解决,因为在做项目过程中你会遇到很多问题,然后你会发现恰当的设计模式能很好的解决你这个问题,这个时候你才会觉得前人总结的设计模式的伟大之处。

那么什么是策略模式呢?

        策略模式定了算法族,分别封装起来,让它们之间可以互相替换,
        此模式让算法的变化独立于使用算法的客户。

这是非常官方的描述,单独看这个描述肯定一头雾水,当然我不打算去解释这个定义,我只是简单谈一下我对策略模式的运用,至于完不完全符合策略模式就看实际情况的运用了,理论只是起指导作用,我觉得不可生搬硬套。

当我们需要设计一个类的时候,我们总是需要去思考这个类哪些属性是不需要改变的,哪些属性是需要动态变化的,不需要改变的通常是对象共有的属性,需要变化的则是对象的个性化,这在生活中很常见,我们总是拥有某些共性的同时,保留自己的个性。

让我们假设一个场景,游戏中玩家需要战斗,战斗需要武器,每个玩家是一个个体,你会很容易发现这里的共性是玩家的基本属性,如姓名、年龄等,行为共性则是战斗,玩家都需要战斗,但是战斗使用的武器是不一样的,每个玩家的都不一样,同一个玩家也可能换武器,因此设计的时候不能把武器的实现与玩家绑死。

下面是设计的类图:


策略模式简单谈_第1张图片
类图.png

实现代码例子为:

//Character.java
package com.base;

//玩家战斗不变的基础属性独立出来的超类,可复用
public class Character {
    //玩家的共有的属性
    private String name;
    private int age;
    
    //Weapon是一个接口
    protected Weapon w;
    
    public String getName(){
        return name;
    }
    
    public void setName(String name){
        this.name = name;
    }
    
    public int getAge(){
        return age;
    }
    
    public void setAge(int age){
        this.age = age;
    }
    
    //可以动态的改变武器
    public void setWeapon(Weapon wt){
        this.w = wt;
    }
    
    //玩家都是使用武器战斗,但不同玩家拥有不同的武器
    public void fight(){
        w.useWeaponFight();
    }
    
}

Character是一个超类,通过继承Character实现每个玩家类,你会发现Character的变量成员和方法都没有被绑死,拿fight()方法来谈,不管w接口由谁实现,反正这里调用的就是useWeaponFight()方法,使用武器进行战斗,战斗是每个玩家都必须具备的,这个在这个场景下永远不会改变,同时还提供:

setWeapon(Weapon wt) 

方法,可以动态改变武器,玩家也不会和武器类绑死,这样设计类代码可以具备足够的弹性。

下面其它类的简单实现:

//Weapon.java
package com.base;

//武器接口,通过实现该接口产生不同的武器
public interface Weapon {
    public void useWeaponFight();
}

//StrongWeapon.java
package player;
import com.base.Weapon;

//实现一个强力武器
public class StrongWeapon implements Weapon{
    public void useWeaponFight(){
        System.out.println("Use Strong Power Weapon Fighting!");
    }
}

//StrongPlayer.java
package player;
import com.base.Character;
//产生一个玩家,玩家的武器初始化在构造器中
public class StrongPlayer extends Character{
    public StrongPlayer(){
        this.w = new StrongWeapon();
    }
}
//WeakWeapon.java
package player;
import com.base.Weapon;

//实现一个弱势武器
public class WeakWeapon implements Weapon{
    public void useWeaponFight() {
        System.out.println("Use Weak Weapon Fighting!");
    }
}

//WeakPlayer,java
package player;
import com.base.Character;

public class WeakPlayer extends Character{

    public WeakPlayer() {
        this.w = new WeakWeapon();
    }
}

然后我们来测试一下两个玩家类,测试代码为:

//StartFight.java
import player.*;

public class StartFight {
    public static void main(String[] args){
        StrongPlayer sp = new StrongPlayer();
        sp.fight();
        
        WeakPlayer wp = new WeakPlayer();
        wp.fight();
        
        //切换武器
        sp.setWeapon(new WeakWeapon());
        sp.fight();
        
        //切换强力的武器
        wp.setWeapon(new StrongWeapon());
        wp.fight();
    }
}

输出结果:

Use Strong Power Weapon Fighting!
Use Weak Weapon Fighting!
Use Weak Weapon Fighting!
Use Strong Power Weapon Fighting!

上面只是一个简单的例子,回到策略模式的定义,这里所谓的封装的算法族体现为每个武器类,每个武器类的使用武器的方法useWeaponFight()都是不一样的,这个算法是可以变化的,比如StrongWeapon,假设现在需求改变了该武器的使用方法,那么我们只需要修改StrongWeapon的useWeaponFight()方法就可以了,原来的代码都不用修改,这样就达到了我们设计良好代码的目的啦。

以上为策略模式简单谈的全部内容,接下来会继续学习其他设计模式,也会继续和大家分享,该案例的工程代码也上传至github,有兴趣的同学可以继续扩展。https://github.com/huchangwei/StrategyMode

你可能感兴趣的:(策略模式简单谈)