- 找出应用中可能需要变化之处,把它们独立出来,不要和那些不需要变化的代码混在一起。
换句话说,如果每次新的需求一来,都会使某方面的代码发生变化,那么你就可以确定,这部分代码需要被抽出来,和其他稳定的代码有区别。例如例子中的鸭子会飞 fly()方法,就需要独立出来。(把会变化的部分取出来并封装起来,以便以后可以轻易的改动或扩充此部分,而不影响不需要变化的其他部分)
- 针对接口编程,而不是针对实现编程
委托(delegate)
- 设计模式
- 例子及总结
问题描述:
现在想让鸭子能飞,该如何设计,如果只是在Duck这个类里增加飞的方法,不好,因为继承的子类里有的是玩具鸭子不能飞,如果是单独定义的接口,也不好,每个类实现接口的时候都需要每个都重新编码,飞的方法可能是固定的就那么1-2种,如果每个类都需要重新实现是不好的方法。那么怎么办呢?
将变化的方法抽象出来,比如飞和叫,然后再定义两个单独的类来实现该接口,那然后把这两个类作为自己的实例变量来实现。
- 例子分析
目前我们有一套关于鸭子的游戏,有一个super class Duck, 如下图: 现在… boss说要鸭子可以飞。。。怎么办呢?(要求代码尽量少,并尽量满足oop), 我们知道,最好就是能够“一劳永逸”, 而且满足可复用,就是说,当再有新的方法添加时,比如鸭子可以”泡妞“,对别的不产生影响。
办法1: 在super class 中直接添加 fly()方法, 一劳永逸+可复用都满足,但是。。不满足逻辑,比如,水中的玩具鸭子,它能飞? 它是不可以飞的。即该子类不能继承这个方法。(此处你也可以在不能飞的子类继承时不实现fly()方法,但是,这不是结构在限制你,而是人为限制,人为的行为,就很有可能出现错误)
办法2: 定义一个fly接口,让可以飞的鸭子实现该接口,逻辑上过的去,该飞的可以飞,不该飞的不实现该接口就可以了。也可复用,但是不能做到一劳永逸,因为所有实现该接口的类必须完成该方法,而fly()方法是比较固定的,”煽动翅膀就可以飞了“。 要在每个实现该接口的子类里都定义一般”飞“,很多重复代码。
办法3:
UML: 类图
code:
Duck-Duck/**
* File: SouckBehavior.java
* ----------------------------------
* Do the sock job interface
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:52
*/
public interface SouckBehavior {
public void sock();
}
/**
* File: FlyBehavior.java
* --------------------------------------
* Do the fly job interface
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:46
*/
public interface FlyBehavior {
public void fly();
}
/**
* File: SocuSilence.java
* -------------------------------
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:51
*/
public class SockSilence implements SouckBehavior {
public SockSilence(){
}
public void finalize() throws Throwable {
}
public void sock(){
System.out.println("I am sorry, i can't sock.");
}
}
/**
* File: SockLow.java
* ------------------------
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:50
*/
public class SockLow implements SouckBehavior {
public SockLow(){
}
public void finalize() throws Throwable {
}
public void sock(){
System.out.println("I can sock, low, ^_^");
}
}
/**
* File: SockHigh.java
* ----------------------------------
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:50
*/
public class SockHigh implements SouckBehavior {
public SockHigh(){
}
public void finalize() throws Throwable {
}
public void sock(){
System.out.println("I can sock, High, ^+^.");
}
}
/**
* File: FlyOk.java
* ------------------------------
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:48
*/
public class FlyOk implements FlyBehavior {
public FlyOk(){
}
public void finalize() throws Throwable {
}
public void fly(){
System.out.println("As you know, I can fly.");
}
}
/**
* File: FlyNg.java
* ---------------------------------------
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:47
*/
public class FlyNg implements FlyBehavior {
public FlyNg(){
}
public void finalize() throws Throwable {
}
public void fly(){
System.out.println("I am sorry, I can not fly.");
}
}
/**
* File: SockDuck.java
* -------------------------------
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:49
*/
public class SockDuck extends Duck {
public SockDuck(){
flyBehavior = new FlyNg();
sockBehavior = new SockHigh();
}
public void finalize() throws Throwable {
super.finalize();
}
public void display(){
System.out.println("I am a sock duck, I can sock high.");
}
}
/**
* File: NowayDuck.java
* ---------------------------------
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:49
*/
public class NowayDuck extends Duck {
public NowayDuck(){
flyBehavior = new FlyNg();
sockBehavior = new SockSilence();
}
public void finalize() throws Throwable {
super.finalize();
}
public void display(){
System.out.println("I am a No way duck, i can do nothing.");
}
}
/**
* File: FlyDuck.java
* -------------------------------------
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:46
*/
public class FlyDuck extends Duck {
public FlyDuck(){
flyBehavior = new FlyOk();
sockBehavior = new SockSilence();
}
public void finalize() throws Throwable {
super.finalize();
}
public void display(){
System.out.println("I am a fly duck, So i can fly only.");
}
}
/**
* File: FlyAndSockDuck.java
* ---------------------------------
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:45
*/
public class FlyAndSockDuck extends Duck {
public FlyAndSockDuck(){
flyBehavior = new FlyOk();
sockBehavior = new SockLow();
}
public void finalize() throws Throwable {
super.finalize();
}
public void display(){
System.out.println("I am a Fly and Sock duck, Fly And Sock, i can do it!");
}
}
/**
* File: Duck.java
* --------------------------------
* @author Leon
* @version 1.0
* @created 15-五月-2013 14:16:44
*/
public abstract class Duck {
protected FlyBehavior flyBehavior;
protected SouckBehavior sockBehavior;
public Duck(){
}
public void finalize() throws Throwable {
}
public abstract void display();
public void flyMethod(){
flyBehavior.fly();
}
/**
*
* @param fb
*/
public void setFlyBehavior(FlyBehavior fb){
flyBehavior = fb;
}
/**
*
* @param sb
*/
public void setSockBehavior(SouckBehavior sb){
sockBehavior = sb;
}
public void sockMethod(){
sockBehavior.sock();
}
public void swim(){
System.out.println("I am swimming ! ");
}
}
/*
* File: TestDuck.java
* -------------------------------
* Test class for the duck.
*/
public class TestDuck {
public static void main(String[] args) {
/*
FlyDuck du = new FlyDuck();
du.setSockBehavior(new SockHigh());
du.flyMethod();
du.sockMethod();
*/
FlyAndSockDuck dus = new FlyAndSockDuck();
dus.flyMethod();
dus.sockMethod();
dus.setFlyBehavior(new FlyNg());
dus.setSockBehavior(new SockSilence());
dus.flyMethod();
dus.sockMethod();
}
}