您的位置:首页 > 其它

设计模式-策略模式

2017-11-12 21:48 260 查看
最近在看设计模式,就将自己看的一些心得写下来,做学习笔记。该篇看的是head first 设计模式的一些体会和总结。

策略模式

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

这定义好像很晦涩难懂呀,简单点说,就是将行为变化部分分别封装成接口,然后使用组合接口的方式来实现一个对象的全部行为。(在下面的例子看我之后再回头看定义,我想你会明白的)

设计原则(一):找出应用中可能需要变化之处,把它们独立出来,不要和那些不需要变化的代码混在一起。

其实就是把变化的部分封装起来,不要让其他不变的部分受到影响

设计原则(二):针对接口编程,而不是针对实现编程

针对接口编程的本质是:针对超类型(super)编程

书中对这个原则有详细的解释(感觉这段解释很好):这里所谓的“接口”有多个含义,接口是一个“概念”,也是一直java的interface构造。利用多态,程序可以针对超类型编程,执行时会根据实际状况执行到真正的行为,不会绑死在超类型的行为上。“针对超类型编程”这句话,可以更明确的说成“变量的声明类型应该是超类型,通常是一个抽象类或者是一个接口”,如此,只要是具体实现此超类型的类所产生的对象,都可以指定给这个变量。这也意味着,声明类时不用理会以后执行时的真正对象类型!

下面是简单多态的例子:

a.针对“实现编程”

声明变量“d”为Dog类型(Animal的具体实现),会造成我们必须针对具体实现编程。

Dog d = new Dog;
d.bark();


b.但是针对“接口/超类型编程”的做法如下(正确做法):

我们知道对象是狗,但是现在利用annimal进行多态调用。

Animal animal = new Dog();
animal.makeSound();


设计原则(三):多用组合,少用继承

本质就是:通过多个接口(interface),将行为分类(比如鸭子叫和飞属于2个不同的属性),并实现该接口。这里所说的组合就是不使用继承来实现鸭子的叫和飞属性,而是通过定义接口。然后通过接口将这两个行为组合成鸭子的属性(叫和飞)。

在head first 里面是通过定义FlyBehavior 和 QuckBehavior.每个鸭子都有一个FlyBehavior 和 QuckBehavior,好将飞行和呱呱叫委托给他们代为处理。

代码示例(源自head first 设计模式)

/**
* 鸭子抽象超类
* Created by nana on 2017/11/5.
*/

public abstract class Duck {
public FlyBehavior flyBehavior;//这里声明接口类型的引用变量
public QuackBehavior quackBehavior;

public Duck(){
}

public abstract void display();

public void performFly(){
flyBehavior.fly();//委托给飞行行为类
}

public void performQuck(){
quackBehavior.quack();//委托给呱呱叫行为类
}

//游泳方法是所有鸭子都能实现的行为(且表现相同)
public void swim(){
Log.i("nana","All duck float,even decoys");
}
public void setFlyBehavior(FlyBehavior flyBehavior){
this.flyBehavior = flyBehavior;
}

public void setQuackBehavior(QuackBehavior quackBehavior){
this.quackBehavior = quackBehavior;
}
}


鸭子有2个属性(飞行和呱呱叫)下面是2个属性定义接口

/**
* 飞行行为接口
* Created by nana on 2017/11/5.
*/

public interface FlyBehavior {
void fly();
}


/**
* 叫声接口
* Created by nana on 2017/11/5.
*/

public interface QuackBehavior {
void quack();
}


FlyNoWay 和 FlyWithWings 是对飞行接口 FlyBehavior 的具体实现

/**
* 不会飞的鸭子的飞行行为实现类
*/

public class FlyNoWay implements FlyBehavior {
@Override
public void fly() {
Log.i("nana","FlyNoWay!!!");}
}


/**
* 鸭子用翅膀飞行的飞行实现类
*/

public class FlyWithWings implements FlyBehavior{
@Override
public void fly() {
Log.i("nana","FlyWithWings!!!");
}
}


Quck是对鸭子叫行为的具体实现

public class Quck implements QuackBehavior{

@Override
public void quack() {
Log.i("nana","quack!!!");
}
}


这个MallardDuck是鸭子抽象超类的子类实现

public class MallardDuck extends Duck{

public MallardDuck(){
quackBehavior = new Quck();
flyBehavior = new FlyWithWings();
}

@Override
public void display() {
Log.i("nana","I am a Mallard duck ~~~~~");
}
}


测试类:

public class MainActivity extends AppCompatActivity {

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Duck mallard = new MallardDuck();//这里针对接口编程,多态
mallard.performFly();
mallard.setFlyBehavior(new FlyWithWings());
mallard.performFly();
mallard.performQuck();
mallard.display();
mallard.swim();
}
}


看完代码,我们再来总结一下。使用原则一,将鸭子的行为呱呱叫和飞行行为分离出来。使用原则二和原则三,我们针对超类型编程,鸭子有抽象超类(且超类里持有飞行和叫的引用,并在超类里面使用这个应用进行行为的处理),那么子类鸭子就可以继承超类实现飞行和叫的行为。且可以根据自己实际情况选择会飞或者用翅膀飞的飞行行为,叫的行为也可以有多个,这里没有具体写。

不知道你理解了吗?看一下UML图可能更直观。



上述图片是应用网友的博客中图片,发现该网友对书中例子更全面。可以参考链接:https://www.cnblogs.com/wolf-sun/p/3534573.html

以上是读书笔记,如果有问题。欢迎交流。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  设计模式 headfirst