如何在ActionScript 3.0里使用工厂模式和模板方法模式(Factory and Template Method Patterns)

来源:互联网 发布:淘宝图片助手官方 编辑:程序博客网 时间:2024/06/05 19:40
 

抽象类(Abstract Classes)
        抽象类在工厂和模板方法模式里扮演着重要角色。虽然ActionScript 3.0并没有原生支持它们,但我们依然可以在ActipScript 3.0里套用抽象类和抽象方法的思想。一个抽象类就是一个总被用来继承且永不会直接被实例化的类。它的用途跟接口类相似,但有一个最大的不同之处就是:接口类只定义公有方法的名称而没有具体的执行(就像是只有函数名而没有函数体),但抽象类两者都有。
        抽象类使用抽象方法,这些抽象方法没有具体的执行,只用作占位符。在其它语言,例如 C#、Java,你可以使用 abstract 关键字来声明抽象方法,它的子类必须重写这些抽象方法。但在ActionScript 3.0里并没有 abstract 关键字,所以我们得想办法解决此问题。
        在ActionScript 3.0里使用抽象类,你必须知道两个特别的关键字:override 和 final。子类必须使用override关键字重写抽象类的方法,方法名称必须相同。使用final关键字来声明那些不被子类重写的方法,在模板方法模式里就用到final关键字了。

模板方法
        定义在抽象类里的模板方法包含一些方案,这些方案来用组织抽象方法。请看以下这个抽象类,在initialize()方法里,我们定义 games 初始化的方式:

  1. package com.peachpit.aas3wdp.factoryexample {
  2.         public class AbstractGame {
  3.                 // 模板函数
  4.                 public final function initialize():void {
  5.                         createField();
  6.                         createTeam("red");
  7.                         createTeam("blue");
  8.                         startGame();
  9.                 }
  10.                 public function createField():void {
  11.                         throw new Error("Abstract Method!");
  12.                 }
  13.                 public function createTeam(name:String):void {
  14.                         throw new Error("Abstract Method!");
  15.                 }
  16.                 public function startGame():void {
  17.                         throw new Error("Abstract Method!");
  18.                 }
  19.         }
  20. }

        在上面例子里,initialize() 方法是一个模板函数。它定义了 game 是这样初始化的:首先调用 createField() 方法,然后调用 createTeam() 方法来创建队伍(teams),最后调用 startGame() 方法。然而在这个抽象类里面,这几个函数都没干什么实际的事情。它们是给子类重写的,让子类来决定具体需要做的事情。抽象类都不能被直接实例化,而且抽象类的方法都必须被重写,所以在上面那三个方法里都放置了 throw 语句,这样就可以防止它们被直接调用了。
        接下来我们创建一个 FootballGame 类来继承 AbstractGame 类。这个子类重写抽象类的方法,然后在 initialize() 方法里被调。

  1. package com.peachpit.aas3wdp.factoryexample {
  2.     public class FootballGame extends AbstractGame {
  3.         public override function createField():void {
  4.             trace("Create Football Field");
  5.         }
  6.         public override function createTeam(name:String):void {
  7.             trace("Create Football Team Named " + name);
  8.         }
  9.         public override function startGame():void {
  10.             trace("Start Football Game");
  11.         }
  12.     }
  13. }

        正如你所看见的,FootballGame 类重写了 createField()、createTeam()以及startGame() 方法,使这些方法更适合 football。然而那个 initialize() 方法没有改变。你可以通过这种方式来写一个 BaseballGame和BastketballGame 类。可以像下面代码那样使用 FootballGame 类:

  1. package com.peachpit.aas3wdp.factoryexample {
  2.     import com.peachpit.aas3wdp.factoryexample.FootballGame;
  3.     import flash.display.Sprite;
  4.     public class FactoryExample extends Sprite {
  5.         public function FactoryExample() {
  6.             // Create an instance of FootballGame
  7.             var game:FootballGame = new FootballGame();
  8.             // Call the template method defined in AbstractGame
  9.             game.initialize();
  10.         }
  11.     }
  12. }

执行上面的代码输出如下。你可以看到,重写后的方法会在那个模板函数(initialize())里调用,这个模板函数没有改变。

  1. Create Football Field
  2. Create Football Team Named red
  3. Create Football Team Named blue
  4. Start Football Game

        也许你会疑惑抽象类有什么用,现在你应该明白了一些,这样做的好处,很显然,扩展性强,以后 createField()、createTeam()以及startGame() 内容需要更改,我们只要再创建一个子类就可以,不必修改其他应用代码。



工厂方法
        现在我们拿上文所举的模板方法例子(点击这里)来说一下工厂方法。很普遍的做法是在模板方法里应用工厂方法。
        在上文所说的例子里,createField() 方法并没有返回任何东西;它只输出“Create Football Field”。让我们来改一下它,让它可以创建并返回一个field对象。因为不同的游戏有不同的field类型,所以我们将要创建一个叫 IField 的接口类,所以field类都会实现这个接口。以下这个接口类定义一个 drawField() 方法:

  1. package com.peachpit.aas3wdp.factory {
  2.     public interface IField {
  3.         function drawField():void;
  4.     }
  5. }


        接下来我们新建一个 FootballField 类来实现 IField 接口。FootballField 类定义如下:

  1. package com.peachpit.aas3wdp.factoryexample {
  2.     import com.peachpit.aas3wdp.factoryexample.IField;
  3.     public class FootballField implements IField {
  4.         public function drawField():void {
  5.             trace("Drawing the Football Field");
  6.         }
  7.     }
  8. }

        工厂方法的目的是连接两个或以上分离但相关的类层次结构。在这个例子里,最顶层的是 AbstractGame 类,它有这些子类:FootballGame、BaseballGamet和BastketballGame类。然后第二层的是 IField 接口类,这些类实现这个接口:FootballField、BaseballField和BasketballField类。这个 abstractGame和IField对象是相关的,但它们具体的实例化视它们的子类而定。下图表示了这些类的层次结构:

注意下面的 createField() 和 initialize() 方法,createField() 方法是一个工厂方法,它返回一个实现IField接口的对象。请看:

  1. package com.peachpit.aas3wdp.factoryexample {
  2.  import com.peachpit.aas3wdp.factoryexample.IField;
  3.  public class AbstractGame {
  4.   // 模板方法
  5.   public final function initialize():void {
  6.    var field:IField = createField();
  7.    field.drawField();
  8.    createTeam("red");
  9.    createTeam("blue");
  10.    startGame();
  11.   }
  12.   // 工厂方法
  13.   public function createField():IField{
  14.    throw new Error("Abstract Method!");
  15.   }
  16.   public function createTeam(name:String):void {
  17.    throw new Error("Abstract Method!");
  18.   }
  19.   public function startGame():void {
  20.    throw new Error("Abstract Method!");
  21.   }
  22.  }
  23. }

这个抽象类和里面的模板方法都是匿名的,具体需要执行的事情在子类里定义。请看以下的 FootballField 类:

  1. package com.peachpit.aas3wdp.factory {
  2.  import com.peachpit.aas3wdp.factory.FootballField;
  3.  import com.peachpit.aas3wdp.factory.IField;
  4.  public class FootballGame extends AbstractGame {
  5.   public override function createField():IField {
  6.    return new FootballField();
  7.   }
  8.   public override function createTeam(name:String):void {
  9.    trace("Create Football Team Named " + name);
  10.   }
  11.   public override function startGame():void {
  12.    trace("Start Football Game");
  13.   }
  14.  }
  15. }

运行以上例子,会输出如下:

  1. Drawing the Football Field
  2. Create Football Team Named red
  3. Create Football Team Named blue
  4. Start Football Game

——————————————————————————–
简单工厂
        请注意,厂模式和工厂方法模式不同,以下这个为简单工厂模式:

  1. package com.peachpit.aas3wdp.factoryexample {
  2.  public class GameFactory {
  3.   public static function createGame(gameType:String):IGame {
  4.    switch(gameType){
  5.     case "football":
  6.      return new FootballGame();
  7.     case "baseball":
  8.      return new BaseballGame();
  9.     case "basketball":
  10.     default:
  11.      return new BasketballGame();
  12.    }
  13.   }
  14.  }
  15. }