跳到主要內容

訪問者模式(Visitor)


  • 表示一個作用於某物件結構中的各元素之操作。它使你可以在不改變各元素之類別的前提下,定義作用於這些元素的新操作。
  • 適用於資料結構相對穩定的系統
  • 它把資料結構和作用於結構上的操作之間的耦合解脫開,使得操作集合可以相對自由地演化。
  • 其目的是要把處理從資料結構分離出來。
  • 其缺點係不易增加新的資料結構
using System;
using System.Collections.Generic;

namespace DPExample
{
    // 狀態
    abstract class Action
    {
        protected string name;
        protected string Name
        {
            get { return name; }
            set { name = value; }
        }

        // 得到男人結論或反應
        public abstract void GetManConclusion(Man concreteElementA);

        // 得到女人結論或反應
        public abstract void GetWomanConclusion(Woman concreteElementB);
    }

    // 人
    abstract class Person
    {
        protected string name;
        protected string Name
        {
            get { return name; }
            set { name = value; }
        }

        // 接受
        public abstract void Accept(Action visitor);
    }

    // 具體狀態
    class Success : Action
    {
        public override void GetManConclusion (Man concreteElementA)
        {
            Console.WriteLine ("{0}{1}時,背後多半有一個偉大的女人。", concreteElementA.GetType().Name, this.GetType().Name);
        }

        public override void GetWomanConclusion(Woman concreteElementB)
        {
            Console.WriteLine ("{0}{1}時,背後多半有一個不成功的女人。", concreteElementB.GetType().Name, this.GetType().Name);
        }
    }

    class Failing : Action
    {
        public override void GetManConclusion (Man concreteElementA)
        {
            Console.WriteLine ("{0}{1}時,悶頭喝酒,誰也不用勸。", concreteElementA.GetType().Name, this.GetType().Name);
        }

        public override void GetWomanConclusion(Woman concreteElementB)
        {
            Console.WriteLine ("{0}{1}時,眼淚汪汪,誰也勸不了。", concreteElementB.GetType().Name, this.GetType().Name);
        }
    }

    class Amativeness : Action
    {
        public override void GetManConclusion (Man concreteElementA)
        {
            Console.WriteLine ("{0}{1}時,凡事不懂也要裝懂。", concreteElementA.GetType().Name, this.GetType().Name);
        }

        public override void GetWomanConclusion(Woman concreteElementB)
        {
            Console.WriteLine ("{0}{1}時,遇事懂也裝作不懂。", concreteElementB.GetType().Name, this.GetType().Name);
        }
    }

    class Marriage : Action
    {
        public override void GetManConclusion(Man concreteElementA)
        {
            Console.WriteLine ("{0}{1}時,感慨道:戀愛遊戲終結時,'有妻徒刑' 遙無期。", concreteElementA.GetType().Name, this.GetType().Name);
        }

        public override void GetWomanConclusion(Woman concreteElementB)
        {
            Console.WriteLine ("{0}{1}時,欣慰曰:愛情長跑路漫漫,婚姻保險保平安。", concreteElementB.GetType().Name, this.GetType().Name);
        }
    }

    // 男人
    class Man : Person
    {
        public override void Accept (Action visitor)
        {
            visitor.GetManConclusion(this);
        }
    }

    // 女人
    class Woman : Person
    {
        public override void Accept (Action visitor)
        {
            visitor.GetWomanConclusion(this);
        }
    }

    // 物件結構
    class ObjectStructure
    {
        private IList elements = new List();

        public void Attach (Person element)
        {
            elements.Add (element);
        }

        public void Detach (Person element)
        {
            elements.Remove (element);
        }

        public void Display (Action visitor)
        {
            foreach (Person e in elements)
                e.Accept (visitor);
        }
    }
}
using System;

namespace DPExample
{
 public class Program
 {
  static void Main()
        {
            ObjectStructure o = new ObjectStructure();
            o.Attach (new Man());
            o.Attach (new Woman());

            Success v1 = new Success();
            o.Display (v1);

            Failing v2 = new Failing();
            o.Display (v2);

            Amativeness v3 = new Amativeness();
            o.Display (v3);

            Marriage v4 = new Marriage();
            o.Display (v4);
  }

 }
}

留言

這個網誌中的熱門文章

用 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 ...

[轉貼]MySQL 交易功能 Transaction 整理 (XYZ的筆記本)

全文轉貼自: http://xyz.cinc.biz/2013/05/mysql-transaction.html   (XYZ的筆記本) ---------------------------------------------------------------------------------------------------------- 資料庫的交易(Transaction)功能,能確保多個 SQL 指令,全部執行成功,或全部不執行,不會因為一些意外狀況,而只執行一部份指令,造成資料異常。 MySQL 常用的兩個資料表類型:MyISAM、InnoDB, MyISAM  不支援交易功能,所以以下的整理,均是針對 InnoDB  而言。 交易功能4個特性 (ACID)  Atomicity (原子性、不可分割):交易內的 SQL 指令,不管在任何情況,都只能是全部執行完成,或全部不執行。若是發生無法全部執行完成的狀況,則會回滾(rollback)到完全沒執行時的狀態。 Consistency (一致性):交易完成後,必須維持資料的完整性。所有資料必須符合預設的驗證規則、外鍵限制...等。 Isolation (隔離性):多個交易可以獨立、同時執行,不會互相干擾。這一點跟後面會提到的「隔離層級」有關。 Durability (持久性):交易完成後,異動結果須完整的保留。 開始進入交易模式 SQL 指令: START TRANSACTION  或  BEGIN 結束交易模式 交易完成:使用  COMMIT  儲存所有變動,並結束交易。 交易過程異常:使用  ROLLBACK  回滾,取消交易,還原到未進行交易的狀態。(若交易過程連線中斷,沒 COMMIT 提交的變更,亦會如同執行 ROLLBACK 取消交易) 儲存點 (SAVEPOINT) 交易過程中,可標示多個不同的儲存點,有需要時可 ROLLBACK 到某個儲存點。 建立儲存點: SAVEPOINT 名稱 刪除儲存點: RELEASE SAVEPOINT 名稱 ROLLBACK 到某個儲存點: ROLLBACK TO SAVEPOINT 名稱 如...