PHP物件導向之訪問者模式+組合模式

來源:互聯網
上載者:User

標籤:script   get   com   diff   傳遞   arc   繼承   組合模式   roo   

  因為原文中延續了組合模式的程式碼範例來講訪問者模式 所以這裡就合并一起來複習了。但主要還是講訪問者模式。顧名思義這個模式會有一個訪問者類(就像近期的熱播劇“人民的名義”中的檢查官,跑到到貪官家裡調查取證,查實後就定罪),被訪問者類調用訪問者類的時候會將自身傳遞給它使用。直接看代碼:

//被訪問者基類

abstract class Unit {
  abstract function bombardStrength();  //擷取單位的攻擊力
  

  //這個方法將調用訪問者類,並將自身傳遞給它
  function accept(ArmyVisitor $visitor){
    $method = "visit" . get_class($this);
    $visitor->$method($this);      //調用訪問者類的方法,這裡使用了 "visit" . get_class($this) 組成了方法的名稱
  }
  

  //按原文的說法是設定一個深度,雖然之後會有調用但這個方法對於理解這個模式不重要可以不用管他(原文範例程式碼中經常有些跟完整模式原理沒太多關係的代碼)
  protected function setDepth($depth){
    $this->depth = $depth;
  }

  function getDepth(){
    return $this->depth;
  }
}

 

//弓箭手
class Archer extends Unit{
  function bombardStrength(){
    return 4;
  }
}

//雷射炮

class LaserCannonUnit extends Unit{
  function bombardStrength(){
    return 44;
  }
}

//騎兵

class Cavalry extends Unit{
  function bombardStrength(){
    return 2;          //騎兵的攻擊力居然比弓箭手低?

  }
}

 

//用於組合繼承了unit類的執行個體,並讓Army和TroopCarrier類繼承removeUnit和addUnit方法,不放基類是因為上述的三個類已經是最小單位了不是一個軍事集團removeUnit和addUnit方法對他們沒用。

abstract class CompositeUnit extends Unit{
  private $units = array();    //存放任何繼承了unit 類的執行個體

  function getComposite(){   //這個方法主要用於判斷當前執行個體是否是一個 CompositeUnit 類
    return $this;
  }

  protected function units(){
    return $this->units;
  }

  function removeUnit(Unit $unit){    //刪除一個軍事單位
    $this->units = array_udiff(
      $this->units,array($unit),

      function($a,$b){return ($a === $b)?0:1;}

    );  
  }

  function addUnit(Unit $unit){        //添加一個軍事單位
    if(in_array($unit,$this->units,true)){
      return;
    }
    $unit->setDepth($this->depth + 1);  
    $this->units[] = $unit;
  }

  function bombardStrength(){
    $ret = 0;
    foreach($this->units as $unit){
      $ret +=$unit->bombardStrength();
    }
    return $ret;
  }

  function accept(Armyvisitor $visitor){    //調用訪問者
    parent::accept($visitor);        //調用基類的accept方法,在第一個用戶端代碼條用裡將會儲存軍事集團整體的一個資訊
    foreach($this->units as $thisunit){   //調用軍事單位accept方法,在第一個用戶端代碼條用裡將會儲存其中每一個軍事單位的資訊
      $thisunit->accept($visitor);
    }
  }
}

 

//軍隊

class Army extends CompositeUnit {

}

//艦隊

class TroopCarrier extends CompositeUnit {

}

 

//訪問者類

abstract class ArmyVisitor{
  abstract function visit(Unit $node);  //訪問者要執行的商務邏輯
  function visitArcher(Archer $node){  //其實我覺得對於理解來說這個抽象類別有一個抽象方法visit()就夠了,原文還多出下面這些方法來繞個圈調用visit

    //...... 
    $this->visit($node);
  }

  function visitCavalry(Cavalry $node){

    //.......
    $this->visit($node);
  }

  function visitLaserCannonUnit(LaserCannonUnit $node){

    //......
    $this->visit($node);
  }

  function visitTroopCarrierUnit(Cavalry $node){

    //......
    $this->visit($node);
  }

  function visitArmy(Cavalry $node){

    //......
    $this->visit($node);
  }
}

//這個訪問者類主要用於擷取並儲存被訪問者對象的資訊
class TextDumpArmyVisitor extends ArmyVisitor {
  private $text = "";
  function visit(Unit $node){
    $ret = "";
    $pad = 4 * $node->getDpth();
    $ret .= sprintf("%{$pad}s","");
    $ret .=get_class($node).": ";
    $ret .= "bombard: " . $node->bombardStrength() . "\n";
    $this->text .=$ret;
  }

  function getText(){
    return $this->text;
  }
}

//用於向每個對象徵稅的訪問者類,用戶端代碼2中將會調用
class TaxCollectionVisitor extends ArmyVisitor{
  private $due=0;
  private $report ="";

  function visit(Unit $node){
    $this->levy($node,1);
  }

  function visitArcher(Archer $node){    //複寫了父類的方法,對於不同的單位徵收不同的稅
    $this->levy($node,2);
  }

  function visitCavalry(Cavalry $node){
    $this->levy($node,3);
  }

  function visitTroopCarrierUnit(TroopCarrierUnit $node){
    $this->levy($node,5);
  }

  private function levy(Unit $unit,$amount){        //主要的商務邏輯
    $this->report .= "Tax levied for" . get_class($unit);
    $this->report .= ": $amount\n";
    $this->due +=$amount;
  }

  function getReport(){
    return $this->report;
  }

  function getTax(){
    return $this->due;
  }
}


//用戶端代碼1(擷取並輸出每個對象的一些資訊)
class UnitScript {
  static function joinExisting(Unit $newUnit,Unit $occupyingUnit){
    $comp;
    if(!is_null($com = $occupyingUnit->getComposite())){
      $comp->addUnit($newUnit);
    } else {
      $comp = new Army();
      $comp->addUnit($occupyingUnit);
      $com->addUnit($newUnit);
    }
    return $comp;
  }
}

 

$main_army = new Army();
UnitScript::joinExisting(new Archer(),$main_army);
UnitScript::joinExisting(new LaserCannonUnit(),$main_army);
UnitScript::joinExisting(new Cavalry(),$main_army);

$textdump = new TextDumpArmyVisitor();
$main_army->accept($textdump);
print $textdump->getText();

 

//用戶端代碼2(對每個對象徵稅,最後輸出總共徵收了多少)
$main_army = new Army();
UnitScript::joinExisting(new Archer(),$main_army);
UnitScript::joinExisting(new LaserCannonUnit(),$main_army);
UnitScript::joinExisting(new Cavalry(),$main_army);
$taxcollector = new TaxCollectionVisitor();
$main_army->accept($taxcollector);
print $taxcollector->getTax();

    //上述的代碼因為太懶沒測試,抱歉! 感興趣的朋友就自己運行調試一下吧!

PHP物件導向之訪問者模式+組合模式

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.