It’s our winter holiday now, and I spend many times in playing games. In last week, I soaked myself in <Prince of Persia: the sands of time>. I love this game, but it’s a pity for me that when I fight to the monsters, I can’t control the heroine. Because of that sometimes I need her help to shot somebody exactly not the other one. OK, I’m crazy Design Patterns in ActionScript-Visitor Controlling two roles will increase the difficulty of manipulation, and it’s not a RTS game.

 

In RTS game, such as Warcraft, you can control many units. When your team attacks someone, the team members will use their own skills. Maybe the Dwarven Sniper will use his gun to shot, while the Mountain King uses his hammer. So, in action script, we may express these in this way.

  1. Var team : Array = new Array () ;
  2.  
  3. team . push ( new   DwarvenSniper ()) ;
  4.  
  5. team . push ( new   MountainKing ()) ;
  6. team . push ( new   Priest ()) ;
  7.  
  8. function   attack ( team : Array ) : void
  9. {
  10. for ( var   i : int = 0 ; i < team . length ; i ++ )
  11. {
  12.  
  13. If   ( team [ i ] instanceof DwarvenSniper )
  14.  
  15. DwarvenSniper ( Team [ i ]) . gunShot () ;
  16.  
  17. Else   if ( team [ i ] instanceof MountainKing )
  18.  
  19. MountainKing ( Team [ i ]) . hammerShot () ;
  20.  
  21. Else   if ( team [ i ] instanceof Priest )
  22.  
  23. Priest ( Team [ i ]) . priestHit () ;
  24.  
  25. }
  26. }

Note: the above code is directly type in Word, so don’t try to complier it, it may contain many grammar mistakes Design Patterns in ActionScript-Visitor

Now, take a look at the attack function. When you pass the team array into it, it may works well. Actually, the team member maybe more than ten, eh, I mean the type of members. So, we need to distinguish them in the iteration. This will cause many if-else. And this is the “bad smell” of our code Design Patterns in ActionScript-Visitor

Here, our problem is that, in an array that may contains many objects, and the action it takes depends on the object type. Further more, the object type is fixed, and the operations of each type are also known, just as the units in Warcraft. What we want to do is organize their basic operations to form a series operation, such as a team in Warcraft to attack the creatures with their normal skills, or attack the buildings with siege skills.

  1. Function creatureAttack ( team : Array ) : void
  2.  
  3. {
  4.  
  5. If ( ……… )
  6.  
  7. //Using it’s skill here
  8.  
  9. Else   if ( ………… )
  10.  
  11. .
  12.  
  13. .
  14.  
  15. .
  16.  
  17. .
  18.  
  19. Else   if ( ……….. )
  20.  
  21. //Using it’s own skill here
  22.  
  23. }
  24.  
  25. Function   buildingAttack ( team : Array ) : void
  26.  
  27. {
  28.  
  29. ….. //the same if-else clauses with different skills
  30.  
  31. }

Actually, our aim is to refactoring the if-else clauses. And that’s what the Visitor pattern does. The intent is as follows.

Represent an operation to be performed on the elements of an object structure. Visitor lets you define a new operation without changing the classes of the elements on which it operates.

–By GOF BOOK

And here is the static diagram of this pattern from the GoF book.

Design Patterns in ActionScript-Visitor

In the Warcraft example, the member is corresponding to the Element, eh, one for each concrete element. And the array is the ObjectStructure. The attacks is the visitor, the concrete visitor is the creature attack and building attack.

So, we can refactor the code by this pattern, let all the members implements the Element interface, and let the attacks implements the Visitor interface. And in the iteration, we can do it in this way.

  1. Var creatureAttack : AttackVisitor = new creatureAttack () ;
  2.  
  3. For ( var   i : int = 0 ; I < team . length ; i ++ )
  4. ( teamElement ) team [ i ] . accept ( creatureAttack ) ;

Here, one accept method replace all the if-else clauses and the concrete operations. Eh, you should implement every accept method in the concrete element like this.

  1. Function accept ( visitor : AttackVisitor ) : void
  2. {
  3. Visitor . visitElementX ( this ) ;
  4. }

If you’re interested in this pattern, find more information by Google Design Patterns in ActionScript-Visitor And you can take a look at the example code in the attach file. Design Patterns in ActionScript-VisitorDownload Full Project

相关文章: