跳到主要內容

組合模式(Composite)


  • 將物件組合樹形結構以表示「部份-整體」的層次結構。組合模式使得用戶對單個物件和組合物件的使用具有一致性。
  • 當發現需求中是表現部份與整體層次的結構時,以及你希望用戶可以忽略組合物件與單個的不同,統一地組合結構中的所有物件時,就應該考慮用組合模式。

基本程式碼

using System;
using System.Collections.Generic;

namespace DPExample
{

 // 組合中的物件宣告介面,實現所有類別共有介面的預定行為 //
 abstract class Component
 {
  protected string name;

  public Component (string name)
  {
   this.name = name;
  }

  public abstract void Add (Component c);
  public abstract void Remove (Component c);
  public abstract void Display (int depth);
 }

 // 葉節點
 class Leaf : Component
 {
  public Leaf (string name) : base(name)
  {}

  public override void Add (Component c)
  {
   Console.WriteLine ("Cannot add to a leaf");
  }

  public override void Remove (Component c)
  {
   Console.WriteLine ("Cannot remove from a leaf");
  }

  public override void Display (int depth)
  {
   Console.WriteLine (new String('-', depth) + name);
  }
 }

 // 枝節點
 class Composite : Component
 {
  private List children = new List();

  public Composite (String name) : base(name)
  {}

  public override void Add (Component c)
  {
   children.Add(c);
  }

  public override void Remove (Component c)
  {
   children.Remove (c);
  }

  public override void Display (int depth)
  {
   Console.WriteLine (new String('-', depth) + name);

   foreach (Component component in children)
    component.Display (depth + 2);
  }
 }
}
using System;

namespace DPExample
{
 public class Program
 {
  static void Main ()
  {
   Composite root = new Composite("root");
   root.Add (new Leaf("Leaf A"));
   root.Add (new Leaf("Leaf B"));

   Composite comp = new Composite("Composite X");
   comp.Add (new Leaf("Leaf XA"));
   comp.Add (new Leaf("Leaf XB"));

   root.Add (comp);

   Composite comp2 = new Composite("Composite XY");
   comp2.Add (new Leaf("Leaf XYA"));
   comp2.Add (new Leaf("Leaf XYB"));

   comp.Add (comp2);

   root.Add (new Leaf("Leaf C"));

   Leaf leaf = new Leaf("Leaf D");
   root.Add (leaf);
   root.Remove (leaf);

   root.Display(1);

  }

 }
}


實例程式碼

using System;
using System.Collections.Generic;

namespace DPExample
{
 abstract class 公司
 {
  protected string name;

  public 公司 (string name)
  {
   this.name = name;
  }

  public abstract void Add (公司 c);
  public abstract void Remove (公司 c);
  public abstract void Display (int depth);
  public abstract void LineOfDuty();
 }

 class 具體公司 : 公司
 {
  private List<公司> children = new List<公司>();

  public 具體公司 (string name) : base(name) {}

  public override void Add (公司 c)
  {
   children.Add (c);
  }

  public override void Remove (公司 c)
  {
   children.Remove (c);
  }

  public override void Display (int depth)
  {
   Console.WriteLine (new String ('-', depth) + name);

   foreach (公司 component in children)
   {
    component.Display (depth + 2);
   }
  }

  // 履行職責
  public override void LineOfDuty ()
  {
   foreach (公司 component in children)
   {
    component.LineOfDuty();
   }
  }
 }

 class 人力資源部 : 公司
 {
  public 人力資源部 (string name) : base (name) {}
  public override void Add (公司 c) {}
  public override void Remove (公司 c) {}
  public override void Display (int depth)
  {
   Console.WriteLine (new String ('-', depth) + name);
  }

  public override void LineOfDuty()
  {
   Console.WriteLine ("{0} 員工招聘教育訓練管理", name);
  }
 }

 class 財務部 : 公司
 {
  public 財務部 (string name) : base (name) {}
  public override void Add (公司 c) {}
  public override void Remove (公司 c) {}
  public override void Display (int depth)
  {
   Console.WriteLine (new String ('-', depth) + name);
  }

  public override void LineOfDuty()
  {
   Console.WriteLine ("{0} 公司財務收支管理", name);
  }
 }
}
using System;


namespace DPExample
{
 public class Program
 {
  static void Main ()
  {
   具體公司 root = new 具體公司("北京總公司");
   root.Add (new 人力資源部("總公司人力資源部"));
   root.Add (new 財務部("總公司財務部"));

   具體公司 comp = new 具體公司("上海華東分公司");
   comp.Add (new 人力資源部("華東分公司人力資源部"));
   comp.Add (new 財務部("華東分公司財務部"));
   root.Add (comp);

   具體公司 comp1 = new 具體公司("南京辦事處");
   comp1.Add (new 人力資源部("南京辦事處人力資源部"));
   comp1.Add (new 財務部("南京辦事處財務部"));
   root.Add (comp1);

   具體公司 comp2 = new 具體公司("杭州辦事處");
   comp2.Add (new 人力資源部("杭州辦事處人力資源部"));
   comp2.Add (new 財務部("杭州辦事處財務部"));
   root.Add (comp2);

   Console.WriteLine ("\n結構圖:");
   root.Display(1);

   Console.WriteLine("\n職責:");
   root.LineOfDuty();
  }

 }
}

留言

這個網誌中的熱門文章

用 C# 批次控制 Word 合併列印

前由 我有全區的電話資料,問題在於我要依不同里別來製作出電話簿。結果如下圖: 單純採用合併列印無法達成我的需求。解決方法係用「功能變數」儲存上一個里別,與現在里別進行比較:若不同,則換頁。不過,這樣功能變數還蠻長的。最後,我還是採用 C# 來解決。 解決方案 用 C# 控制 WORD 中合併列印的「資料來源 Data Source」,給予不同里別的「sqlstatement」。迴圈處理不同的里別即可。但可預見其處理過程會很慢,不過還好,我可以不用在意它,有跑出結果即可。 程式碼 IList<string> areas = new List<string>() { "後壁", "侯伯", "嘉苳", "土溝", "嘉田", "嘉民", "菁豊", "崁頂", "後廍", "墨林", "菁寮", "新嘉", "頂長", "平安", "仕安", "竹新", "新東", "長安", "頂安", "福安", "烏樹" }; string root = @"D:\"; // 根目錄 string data = root + @"\data.docm"; // 資料檔(即資料來源) string template = root + @"\template.docx"; // 已設定好格式與合併欄位的 Word 檔 string output = @"d:\Final"; // 輸出之資料夾 object oMissing = System.Reflection.Missing.Va...

VLC c# 順利編譯

原文網址: http://www.cnblogs.com/haibindev/archive/2011/12/21/2296173.html 原文作者: haibindev 原文標題:c#万能视频播放器 本文的重點在於修正 class VlcPlayer,使其能順利在 VC# Express 2010 .Net Framework 4 下順利編譯。 修正重點在於 CallingConvention = CallingConvention. StdCall 改成 CallingConvention = CallingConvention. Cdecl using System; using System.Runtime.InteropServices; using System.Security; using System.Text; namespace VlcDotNet { class VlcPlayer { private IntPtr libvlc_instance_; private IntPtr libvlc_media_player_; private double duration_; public VlcPlayer(string pluginPath) { string plugin_arg = "--plugin-path=" + pluginPath; string[] arguments = { "-I", "dummy", "--ignore-config", "--no-video-title", plugin_arg }; libvlc_instance_ = LibVlcAPI.libvlc_new(arguments); libvlc_media_player_ = LibVlcAPI.libvlc_media_player_new(libvlc_instance_); } public ...

[Symfony+Doctrine] 透過非 Id 來使用 Pessimistic Lock

根據 文件 ,Doctrine 有 Pessimistic Lock,又分為兩種: LockMode::PESSIMISTIC_WRITE:對應至 MySQL 的 Select FOR UPDATE LockMode::PESSIMISTIC_READ:對應至 MySQL 的 Select LOCK IN SHARE MODE 差別在於 LOCK IN SHARE MODE 會將在 row 的資料鎖定(row-level lock),在非同一交易(Transaction)下,不給寫入,其他交易可以讀取, 且可以繼續 Select LOCK IN SHARE MODE 。而 FOR UPDATE 不僅鎖定該資料,在非同一交易下,不給寫入,其它交易可以讀取, 但不能 Select LOCK IN SHARE MODE 。MySQL 文件有更詳細的比較與情境使用的說明,參考 網址 。 現在問題是,我們要完全採用 ORM 來處理資料。Doctrine 的文件提到 EntityManager::find 可以採用 Pessimistic Lock, 但 find 是透過 id 來處理 。而其他 find 系列函數(包括:findAll, findBy, findOneBy)皆不支援 LockMode。 因此,勢必要有方法來「透過非 id 來使用 Pessimistic Lock」。透過查看原始碼,簡單的方法是有的,解法之範例如下: 19 public function depositAction() 20 { 21 22 $em = $this->getDoctrine()->getManager(); 23 24 $em->transactional(function ($em) { 25 $entityName = 'AcmeTrainingBundle:Account'; 26 $lockMode = LockMode::PESSIMISTIC_READ; 27 $orderBy = null; 28 $...