2012年11月28日 星期三

ASP.NET Page-Life-Cycle

前言:
開發 ASP.NET Web Form 的工程師對於網頁生命週期(Page Life Cycle)一定要熟記在心中,也要知道每個循環中的會發生的個事件的主要目的,才不會在需求要修改或是出現Bug時不知道該往哪裡找問題。不過,講是這樣講其實ASP.NET的網頁生命週期需要注意的細節是滿多的,而且一個週期內會發生的事件可能比你想像中的還要多。

內文:
一般來說網頁生命週期會經過 "網頁要求"=>"開始"=>"初始化"=>"載入"=>"回傳事件處理"=>"呈現"=>"卸載"

階段 說明
網頁要求 在網頁生命週期開始前會發生網頁要求。當使用者要求網頁時,ASP.NET 會判斷是否需要剖析和編譯網頁 (因此開始網頁的生命週期),或是可以在不執行網頁的情況下,傳送網頁的快取版本做為回應。
開始 在開始階段,會先設定網頁屬性,如 Request 與 Response。在這個階段中,網頁也會判斷要求是否為回傳或是新的要求,然後設定 IsPostBack 屬性。網頁同時也會設定 UICulture 屬性。
初始化 在網頁初始化期間,可以使用網頁上的控制項,並且設定每個控制項的 UniqueID 屬性。如果適用,也會將主版頁面與佈景主題套用到網頁。如果目前的要求是回傳,則尚未載入回傳資料,並且控制項屬性值並未還原至檢視狀態提供的值。
載入 在載入期間,如果目前的要求是回傳,就會使用從檢視狀態和控制項狀態復原的資訊載入控制項屬性。
回傳事件處理 如果是回傳要求,會先呼叫控制項事件處理常式,然後才會呼叫所有驗證程式控制項的 Validate 方法,以設定個別驗證程式控制項與網頁的 IsValid 屬性。
呈現 在呈現前,會儲存網頁和所有控制項的檢視狀態。在呈現階段,網頁會呼叫每個控制項的 Render 方法,藉此提供文字寫入器將其輸出寫入網頁之 Response 屬性的 OutputStream 物件。
卸載 完整呈現網頁之後,會引發 Unload 事件,然後傳送至用戶端予以捨棄。此時將會執行網頁屬性 (如 Response 與 Request) 的卸載及清除工作。
資料來源 表一:MSDN 一般網頁生命週期階段 Page Life Cycle

而在這樣的生命週期階段中會有數十個事件依序發生,而不論是作為網頁容器的 Page 類別或是使用者控制項(User Control)或是伺服器控制項(Server Constrol)幾乎都有相對應的事件。
Page 頁面是每個 Web Form 都會繼承的類別,依據需求上面也可以放置自行撰寫的使用者控制項(User Control),或是裡面也可以直接擺上 Server Control,因此我們可以把這個類別是為一個容器,裡面可以擺放很多顯示用的控制項,而這個控制項裡會有很多事件會在一個 request 進來後依序引發,以下就是網頁生命週期的「生命週期事件」:

OnPreInit:
這個事件會在初始化之前就引發,通常會拿來檢查 PostBack 屬性、建立或是重建動態控制項、動態設定主版頁面、動態設定 Theme 屬性、讀取貨設定設定黨屬性值。

OnInit:
初始化所有控制項並套用任何面板設定後引發。個別控制項的 OnInit 事件會在容器前執行。可以在這個時候讀取或是初始化控制項的屬性。

OnInitComplete:
發生在網頁初始化結束時。

OnPreLoad:
這邊會於載入本身以及所包含的控制項的 ViewState,以及處理 request 執行個體中所附的回傳資料後引發。
若有自定驗證的話通常我會在這邊來進行,就不用載入太多東西。

OnLoad:
這邊容器會呼叫本身的 OnLoad 方法,然後以遞迴的方式對每一個控制項執行相同的動作。(因此控制項的 Load 會比較晚執行)通常我會於此設定控制項中的屬性,以及重建資料庫連線。

OnLoadComplete:
於 OnLoad 事件處理完畢後引發。

OnPreRender:
容器會呼叫自己的 PreRender 事件,然後也是以遞迴的方式對每一個控制項執行相同的動作。這個事件可以在呈現(Render)階段前對控制項的屬性做最後的變更。
通常我會在這邊把需要的 javascript 或 CSS 給引入或是對顯示邏輯進行修改。

OnSaveStateComplete:
這個事件是在儲存網頁所有控制項的 ViewState 後引發,因為是在呼叫Render方法之前,因此所做的任何變更依舊會影響控制項的呈現,但是因為 ViewState 已經儲存了,所以下一次的 PostBack 並不會擷取這些變更!
(因此PreRender才會被視為呈現前最後一個可以變更控制項的事件,因為此時變更後會被儲存下來!)

Render:
這不是個事件...在呈現這個階段每個控制項都會呼叫自己的Render方法來產生各自的HTML Tag。

OnUnload:
會先由控制項引發,控制項都引發完畢後才是容器引發。通常在這邊會對頁面使用到的特定資源進行釋放,像是與資料庫的連線或是關閉開啟的檔案。

以上資料來源 MSDN 生命週期事件 + 我的胡說八道

當然上面這些說明的文件看完之後還是需要手動來驗證一下才比較有感覺!
就讓我們來驗證一下吧!

驗證:

實驗一:
1)首先我們先新增一個空白ASP.NET Web Form專案,並且新增項目選擇Web Form,命名為Index.aspx。

2)打開 Package Manager Console 安裝NLog Install-Package NLog Install-Package NLog.Config
這邊安裝NLog是為了用來記錄每個事件的觸發順序使用。

3)修改NLog.config
<targets>
    <target xsi:type="File" name="test" fileName="${basedir}/logs/${shortdate}.log"
            layout="${longdate} ${message}"/>
  </targets>

  <rules>
    <logger name="test" minlevel="Trace" writeTo="test" />
  </rules>

4)在Index.aspx.cs中寫下日誌
using System;
using NLog;
public partial class Index : System.Web.UI.Page
{
    private Logger _logger = LogManager.GetLogger("test");

    protected override void OnPreInit(EventArgs e)
    {
        _logger.Trace("OnPreInit event execute!");
        base.OnPreInit(e);
    }

    protected override void OnInit(EventArgs e)
    {
        _logger.Trace("OnInit event execute!");
        base.OnInit(e);
    }

    protected override void OnInitComplete(EventArgs e)
    {
        _logger.Trace("OnInitComplete event execute!");
        base.OnInitComplete(e);
    }

    protected override void OnPreLoad(EventArgs e)
    {
        _logger.Trace("OnPreLoad event execute!");
        base.OnPreLoad(e);
    }

    protected override void OnLoad(EventArgs e)
    {
        _logger.Trace("OnLoad event execute!");
        base.OnLoad(e);
    }

    protected override void OnLoadComplete(EventArgs e)
    {
        _logger.Trace("OnLoadComplete event execute!");
        base.OnLoadComplete(e);
    }

    protected override void OnPreRender(EventArgs e)
    {
        _logger.Trace("OnPreRender event execute!");
        base.OnPreRender(e);
    }

    protected override void OnPreRenderComplete(EventArgs e)
    {
        _logger.Trace("OnPreRenderComplete event execute!");
        base.OnPreRenderComplete(e);
    }

    protected override void OnSaveStateComplete(EventArgs e)
    {
        _logger.Trace("OnSaveStateComplete event execute!");
        base.OnSaveStateComplete(e);
    }

    protected override void OnUnload(EventArgs e)
    {
        _logger.Trace("OnUnload event execute!");
        base.OnUnload(e);
        _logger.Trace("-------------------");
    }
}

5)按下F5成功執行後直接關起來

6)觀察寫下來的Log檔

這邊我們可以發現頁面的事件順序大致上就是先初始化、載入、呈現最後就是卸載。
但若我們在頁面上面多加一個使用者控制項(User Constrol)呢?

實驗二:

1)在原本的專案中新增一個ascx檔,並且附上一個 Server Control。

2)新增如下內容
<asp:TextBox ID="ucTextBox1" runat="server" 
             OnInit="ucTextBox1_Init"
             OnLoad="ucTextBox1_Load"
             OnPreRender="ucTextBox1_PreRender"
             OnUnload="ucTextBox1_Unload"></asp:TextBox>
public partial class UserControl1 : System.Web.UI.UserControl
{
    private Logger _logger = LogManager.GetLogger("test");

    protected override void OnInit(EventArgs e)
    {
        _logger.Trace("User Control OnInit event execute!");
        base.OnInit(e);
    }

    protected override void OnLoad(EventArgs e)
    {
        _logger.Trace("User Control OnLoad event execute!");
        base.OnLoad(e);
    }

    protected override void OnPreRender(EventArgs e)
    {
        _logger.Trace("User Constrol OnPreRender event execute!");
        base.OnPreRender(e);
    }

    protected override void OnUnload(EventArgs e)
    {
        _logger.Trace("User Constrol OnUnload event execute!");
        base.OnUnload(e);
    }

    protected override void OnDataBinding(EventArgs e)
    {
        _logger.Trace("User Control OnDataBinding event execute!");
        base.OnDataBinding(e);
    }

    #region 伺服器控制項的事件
    protected void ucTextBox1_Unload(object sender, EventArgs e)
    {
        _logger.Trace("ucTextBox1 Unload event execute!");
    }

    protected void ucTextBox1_PreRender(object sender, EventArgs e)
    {
        _logger.Trace("ucTextBox1 PreRender event execute!");
    }

    protected void ucTextBox1_Load(object sender, EventArgs e)
    {
        _logger.Trace("ucTextBox1 Load event execute!");
    }

    protected void ucTextBox1_Init(object sender, EventArgs e)
    {
        _logger.Trace("ucTextBox1 Init event execute!");
    } 
    #endregion
}

3)按下F5成功執行後直接關起來

4)觀察日誌檔

這邊我們可以觀察到初始化時會是控制項的初始化先進行,然後才是 User Control 的初始化,最後是 Page 的初始化(容器的初始化會比較晚)。
而在載入的時候會先由 Page 先行載入,然後是 User Control 的載入,最後才是伺服器控制項的載入(容器會先行載入)。
而在卸載的時候則是控制項先卸載,然後是 User Constrol,最後才是 Page(容器的卸載的比較晚)。

實驗三:
現在我們在 Page 頁面也新增兩個 button 來觀察看看各控制項的事件引發順序。

1)我們分別在使用者控制項(User Control)的前後增加一個伺服器控制項(Server Control)。現在Index.aspx就會變成下面這樣:
<form id="form1" runat="server">
    <div>
        <asp:Button ID="button1" runat="server" Text="Firstbutton" 
                    OnInit="button1_Init"
                    OnLoad="button1_Load"
                    OnPreRender="button1_PreRender"
                    OnUnload="button1_Unload" />
        <uc:UserControl1 ID="uc1" runat="server" Visible="true" />
        <asp:Button ID="button2" runat="server" Text="Secondbutton" 
                    OnInit="button2_Init"
                    OnLoad="button2_Load"
                    OnPreRender="button2_PreRender"
                    OnUnload="button2_Unload" />
    </div>
</form>

2)在兩個 button 的事件中寫下日誌...方法就跟上面的寫法一樣...
#region 伺服器控制項的事件
    protected void button1_Init(object sender, EventArgs e)
    {
        _logger.Trace("Fisrtbutton Init event execute!");
    }

    protected void button1_Load(object sender, EventArgs e)
    {
        _logger.Trace("Fisrtbutton Load event execute!");
    }

    protected void button1_PreRender(object sender, EventArgs e)
    {
        _logger.Trace("Fisrtbutton PreRender event execute!");
    }

    protected void button1_Unload(object sender, EventArgs e)
    {
        _logger.Trace("Fisrtbutton Unload event execute!");
    }
    protected void button2_Init(object sender, EventArgs e)
    {
        _logger.Trace("Secondbutton Init event execute!");
    }

    protected void button2_Load(object sender, EventArgs e)
    {
        _logger.Trace("Secondbutton Load event execute!");
    }

    protected void button2_PreRender(object sender, EventArgs e)
    {
        _logger.Trace("Secondbutton PreRender event execute!");
    }

    protected void button2_Unload(object sender, EventArgs e)
    {
        _logger.Trace("Secondbutton Unload event execute!");
    } 
#endregion

3)按下F5之後直接關掉觀察日誌檔...

初始化>>
這邊我們可以觀察到初始化的時候會依照控制項出現的順序依序初始化,因此第一個 button 會先被初始化,然後是使用者控制項中的 textbox 初始化,接下來就是使用者控制項的初始化!接下來才是第二個 button 的初始化,最後才是 Page 頁面(基底容器) 的初始化。(個別控制項的初始化會在它的容器前執行)

載入>>
載入時依據MSDN官方說法是容器會先控制項載入,我們來驗證吧~!這邊我們可以看到 Page 頁面的載入確實先觸發了!然後是第一個 button 的載入,接下來是使用者控制項的載入,接著使用者控制項中的 textbox 載入,最後是第二的 button 的載入。載入就一這樣的順序結束了!

呈現(Render)>>
這邊很特別,我們可以看到這個 Render 的順序根本就是頁面從上到下的順序依序呼叫 PreRender,不過也難怪啦~畢竟這邊是要產生 HTML Tag 是最後要呈現在瀏覽器的樣子,會有這樣的行為其實不意外,要是順序不是這樣最後呈現可能也會怪怪的。

卸載>>
卸載這邊則會讓控制項先行卸載,最後才是容器卸載。所以這邊第一個 button 就會先執行卸載,接下來是使用者控制項中的 textbox 卸載,接著是使用者控制項,然後是第二個 button,這些控制項都卸載後最後才是 Page 頁面的卸載。

結語:
在撰寫 ASP.NET Web Form 時了解網頁生命週期是不可或缺的知識,他可以幫助你理解 Web Form 架構時的一些眉角,能夠在正確的時間點做正確的事情。雖然現在在強大的 javascript library 或 framework 的幫助下,已經讓網頁的使用者體驗愈趨近 desktop 導致 Web Form 的式微,但是偶而還是會有需要使用到的時候,到時候就會需要了解這方面的知識了。

參考資料:
MSDN
ASP.NET Page Life Cycle Overview
dotblog
In 91 [ASP.NET]Page Lify Cycle整理
m@rcus學習筆記 [.NET]ASP.NET網頁生命週期Page Life-Cycle

2012年11月20日 星期二

Entity Framework Migrations - 入門使用

前言:
使用 Entity Framework 的 Code First 來做資料存取層與既存資料庫溝通時多少都會遇到資料庫 Schema 變更的時候,只是這個時候若是有用到 Code First 來新增表格的話就會遇到像上一篇一樣的問題;執行時期使用資料庫的話會引發錯誤訊息,而且會通知因為資料庫的 Schema 變更需要使用 Migrations 來解決問題。因此這篇文章就是要來說明一般來說使用 Code First 來開發遇到資料庫 Schema 要變更時的正規作法應該怎麼做。

內文:
我們就用上一篇所生出來的資料庫來實作看看吧~


不過為了方便起見,我們就直接先把 Products 這個資料表先刪除,我們來看看用 Code First 的方式開發的話若要新增一個資料表的正規方式是什麼?(這邊正規也只是我在講而已,只是這麼做的話可以免除之後的麻煩)
首先我們要先打開套件管理員的命令列視窗「Tools => Library Package Manager => Package Manager Console」,然後要用 Migrations 的話就要先把功能打開,需要下這樣的指令:「PM >> Enable-Migrations」,成功的同時我們的專案會新增一個Migrations資料夾,並且會給你一個 Configrations.cs 類別,若沒有要使用自動Migrations的話就可以先不用動這個類別的內容。接下來因為需求異動的關係我們需要多一個 Products 資料表來儲存產品的資訊。Products 的資料表如下
/// 
/// 產品的實體。
/// 
[Table("Products")]
public class Product
{
    /// 
    /// 產品的識別號。
    /// 
    [Key]
    [ScaffoldColumn(false)]
    public int ID { get; set; }
    /// 
    /// 產品的名稱。
    /// 
    public string Name { get; set; }
    /// 
    /// 產品的價格。
    /// 
    public double Price { get; set; }
    /// 
    /// 生產產品的工廠名稱。
    /// 
    public string FactoryName { get; set; }
}

對於資料庫的 Schema 的異動我們可以在 Package Manager Console 中輸入 Add-Migrations xxxxxx
xxxxxx就是新增的類別的名稱,通常會是這次異動的表格、欄位的名稱像是AddProducts之類的。這邊我是用AddTableProducts
using System;
using System.Data.Entity.Migrations;

public partial class AddTableProducts : DbMigration
{
    // Operations to be performed during the upgrade process.
    public override void Up()
    {
    }

    // Operations to be performed during the downgrade process.
    public override void Down()
    {
    }
}
使用 Migrations 時一定要實作這兩個方法,而且內容要互相對應。
什麼意思呢?就是說若今天是要新增一個 Table 的話,那我們就要在 Up() 中放置新增 Table 的程式,而在 Down() 中放置捨棄該 Table 的程式像是底下這樣:
public partial class AddTableProducts : DbMigration
{
    public override void Up()
    {
        // 新增一個資料表就呼叫CreateTable方法,需指名資料表明稱,以及資料表各欄位的資料型態
        CreateTable(
            "dbo.Products",
            c => new
                {
                    ID = c.Int(nullable: false, identity: true),
                    Name = c.String(),
                    Price = c.Double(nullable: false),
                    FactoryName = c.String(),
                })
            .PrimaryKey(t => t.ID);
            
    }
        
    public override void Down()
    {
        // Down 這邊要放是 CreateTable的相反方法,這邊就是 DropTable方法
        DropTable("dbo.Products");
    }
}
寫完之後我們在 Package Manage Console 中下個 Update-Database 指令,順利執行完畢之後我們去資料庫中就可以看到新增後的資料表了~
至於新增一筆欄位我們也可以如法炮製,若我們現在 Products 資料表又多了一筆欄位的需求,要把產品的庫存數量記錄下來,首先我們先在 Model 中修改新的 Products 資料表的 Schema (我們在最後面新增了一個屬性"Quantity"):
/// 
/// 產品的實體。
/// 
[Table("Products")]
public class Product
{
    /// 
    /// 產品的識別號。
    /// 
    [Key]
    [ScaffoldColumn(false)]
    public int ID { get; set; }
    /// 
    /// 產品的名稱。
    /// 
    public string Name { get; set; }

    /// 
    /// 產品的價格。
    /// 
    public double Price { get; set; }

    /// 
    /// 生產產品的工廠名稱。
    /// 
    public string FactoryName { get; set; }

    /// 
    /// 產品於庫存的數量。
    /// 
    public int Quantity { get; set; }
}
好哩~這樣Code就算改好了,接下來我們只要下個確認異動的指令讓資料庫異動就好了~
我們在 Package Manage Console 中鍵入「Update-Database」等待執行完畢我們就可以去資料庫確認一下異動是否成功了(其實只要 Console 沒有吐出例外訊息應該都是沒有問題的)

噹~噹~Great!資料庫果然真的有這麼一個欄位在Products資料表中生出來了!
接下來在執行時期也可以很放心的不會吐出任何例外訊息因為資料庫異動而要你去 Migrations 了。
我們也可以來順便看看使用 Migrations 時會在 Demo 資料庫中的 Products 資料表的系統資料表產生的 _Migrations 資料表的內容。
我們這個範例中總共做了 Migrations 兩次,一次是新增資料表,一次是新增欄位:
嗯~果然在這個資料表中我們也發現了兩筆資料。

小結:
在使用 Entity Framework 的 Code First 時對開發人員真的是很方便,以前需要很麻煩的資料庫與程式碼互相切換,還要學會怎麼寫 SQL 這些麻煩事情,使用 Entity Framework 這種 ORM solution 都可以解決掉。而且這次 EF 所提供的 Migrations 功能又可以讓開發人員直接在程式碼中對資料庫異動,完全不需要離開舒適的開發環境。說真的,現在開發要與資料庫互動而且是 MS SQL 那目前的最佳方案我想就是 EF 的 Code First 了!

2012年11月16日 星期五

Code First with existiong db - 手殘問題

前言:
話說之前那一篇的 Entity Framework 觀念其實還是有很多地方是不大清楚的,像是使用 Code First with existing database 時什麼情況會異動到資料庫?什麼情況下需要做Migrations?什麼情況下會直接吐個 Exception 給你而不是自動幫你多個 Table 或 Column?,所以今天就趁著一點時間來寫一下從接觸至今的處理經驗了...(說的好像經驗老到其實還是個新手)

前置工作:
這次的範例中我們會需要一個無辜的資料庫讓我們(恣意蹂躪)測試。
就讓我們把這個偉大的資料庫命名為...Demo!

唔...其實Products先不用建起來,我們可以來做個實驗看看什麼時候會觸發Migrations...好~建立起這個資料庫之後我們就可以馬上來寫Code玩玩看啦~!
這裡我們就使用 Console Application 就好。我們建立一個名為 EFMigrationDemo 的專案,並在根目錄處新增一個 Model 資料夾裡面放置我們最重要的類別。
在寫 Code 之前我們要先把 EntityFramework 裝起來~就直接打開 Package Manager Console 吧!請在 Console 中直接輸入:
Install-Package EntityFramework 若不放心是否為最新就接著下:
Update-Package EntityFramework 這樣就可以保證你所指定的套件是最新的!
前置工作這樣就算是告一段落了~

寫Code囉:
我們新增一個類別檔案於 Model 資料夾中
using System.Data.Entity;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
namespace EFMigrationDemo.Model
{
    public class Customers // 類別名稱與資料庫中的表格名稱一樣
    {
        // 標記該屬性(在資料庫中為欄位)為 Promary key
        //(若屬性結尾為 ID 依照 Conversion naming 規則 EF 會自己判斷出來)
        [Key]
        public int ID { get; set; }

        // 標記該欄位是不能為Null的
        [Required]
        public string Name { get; set; }

        [Required]
        public string Address { get; set; }

        [Required]
        public string PhoneNumber { get; set; }

        [Required]
        public string Email { get; set; }

        // 標記這個屬性對應到資料庫中的欄位名稱為 NickName 而非 TheNickName
        // 而且該欄位的內容長度不超過50
        [Column("NickName")]
        [MaxLength(50, ErrorMessage = "Your Nick Name too long")]
        public string TheNickName { get; set; }
        
        // 標記這個屬性並不會對應到資料庫中的任何欄位
        [NotMapped]
        public string NameplusNickName { get { return Name + NickName; } }

        // 這是個類別所以還是可以有方法,而且不會被認為是欄位或是預儲程序
        public void PrintName() { Console.WriteLine("My name is {0}", this.Name); }
    }

    // 繼承 DbContext 的類別可以當成是資料庫來進行操作,預設會去 Config 檔中抓取 connectionString 的 section 裡與類別名稱相同的項目,像是:
    // <add name="Demo" connectionString="" providerName="">
    // 而類別名稱其實也可以不用跟資料庫名稱一樣
    public class Demo : DbContext
    {
        // 被 DbSet 包起來的類別會被判定為是一個實體(也就是一個表格)
        // 所以屬性名稱並不會影響到表格名稱
        public DbSet<Customers> Customer { get; set; }
    }
} 

這樣我們的 Model 類別就算準備好了,接下來就是準備呼叫端了~
using EFMigrationDemo.Model;
        
        static void Main(string[] args)
        {
            Program p = new Program();
            p.Run();
            Console.WriteLine("Process Run has been finished");
            Console.Read();
        }

        void Run()
        {
            Console.WriteLine("Code First begining...");
            using (Demo db = new Demo())
            {
                var result = (from p in db.Customers
                              select p).FirstOrDefault();
                if (result != null)
                    Console.WriteLine("Customer Name is {0}", result.NameplusNickName);
            }
        }

執行之後就是會印出來資料庫內的第一筆資料哩~
以上就是很順利的 Code First 運作情況,當然事事不會都這麼順利又美好的,若我Customers類別名稱跟資料庫不同的話呢?像是這樣:
    public class MyCustomers
    {
        [Key]
        public int ID { get; set; }

        [Required]
        public string Name { get; set; }

        [Required]
        public string Address { get; set; }

        [Required]
        public string PhoneNumber { get; set; }

        [Required]
        public string Email { get; set; }

        [Column("NickName")]
        [MaxLength(50, ErrorMessage = "Your Nick Name too long")]
        public string TheNickName { get; set; }
        
        [NotMapped]
        public string NameplusNickName { get { return Name + NickName; } }
    }
    
    
    public class Demos : DbContext
    {
        public DbSet<Mycustomersgt; Customer { get; set; }
    }

重新建置後再次執行Program.cs的內容!Console會印出:
Code First begining...
Process Run has been finished
沒了...但是讓我來看看DB有什麼變化吧
噹~噹~!
恭喜你剛剛用 Code First 新建一個 Table 了耶!等等...這不是你要的效果嗎?! Entity Framework 一臉疑惑的問到
這時候你才發現到你剛剛手殘不小心把Table名稱打錯了,改快把名稱改回來重新建置一次,建置成功!讓我們再Run一次吧~
你被殘酷的拒絕了...不過錯誤訊息很貼心的說你應該要用Migration來整合資料庫,因為已經不大一樣了。停!它怎麼知道資料庫不一樣了?
不要急...讓我們看看這張圖吧你就知道為什麼它會知道資料庫被動過了
喔!什麼時候在系統資料表中新增了一個_MigrationHistory資料表了?!而且裡面還有一筆資料!
唔...這就是用 Code First 異動資料庫時它生出來的東西,在這種情況下要重新 run 時就請用 Migration 整合後再說吧!
不過先讓我們看看另外一種錯誤吧!

當我們在撰寫 Customers 類別時實在太粗心了,不小心多加了一個不屬於這個資料表的欄位(假設屬性名稱叫NotExist)也就是說多了一個屬性,而且也沒有貼上[NotMapped]標籤,就直接很開心的 ctr+Alt+b 然後 F5 就下去了。依據經驗法則是不是會在資料庫中多一個欄位呢?讓我們看看吧~
嗯...是個錯誤訊息,表明了這欄位在 Customers 中找不到,所以放心~資料庫中的 Customers 表格也沒有多一個欄位。

最後(ps.Migration怎麼用就留到下一篇吧),預設上 DbContext 會去找跟類別名稱相同的連線字串名稱,但是其實是可以自訂的,這需要參考Syste.Data.Entity.dll。實際方式像是這樣:
public class Demos : DbContext
    {
        public DbSet<customers> Customer { get; set; }
        
        // 這裡呼叫 DbContext 的建構子並且給予一個字串,這個字串就是所指定的連線字串名稱
        // 若有給定的話就不會用預設的方式去找連線字串名稱了
        public Demos() : base("Demo") { }
    }

呼~這篇累積了好多 bug 沒有修啊,但是我打字有點累了...就下回待續吧 XD

2012年11月8日 星期四

EF with Repository Pattern - 簡易版

情境:
上一篇中我們有使用到一個NorthwindReposity類別來存取資料庫,不過我們也發現到這樣的寫法與呼叫端的耦合度還是有點高。若我要使用的資料所要做的事情是固定的,像是瀏覽產品資訊時的產品資料是從主資料庫中找到產品的資料表並取出指定產品的資料,那麼其實對這個部分來說其實他就只關心產品得資料如何存取和操作而已。一個簡單的想法是,我們讓呼叫端用一個介面來針對他所感興趣的部份去做操作。

實作:
我們用的資料庫依然是北風資料庫,而且和上一篇是一樣的!避免頁面切來切去這邊就直接寫囉~
using System.Data.Entity;
System.ComponentModel.DataAnnotations.Schema;
public class Northwind : DbContext
{
    public DbSet<product> Products { get; set; }

    public DbSet<category> Categories { get; set; }
}
public class Product
{
    public int ProductID { get; set; }
    public string ProductName { get; set; }
    public Decimal? UnitPrice { get; set; }
    public bool Discontinued { get; set; }
    [ForeignKey("Category")]
    public int CategoryID { get; set; }

    public virtual Category Category { get; set; }
}
public class Category
{
    public int CategoryID { get; set; }
    public string CategoryName { get; set; }
    public string Description { get; set; }
    public byte[] Picture { get; set; }

    public virtual ICollection<product> Products { get; set; }
}
接下來就是我們上一篇用來存取資料庫的主要類別,也是我們這次的修改標的物。
public class NorthwindRepository : IDisposable
    {
        private Northwind _db;
        
        public NorthwindRepository()
        {
            _db = new Northwind();
        }

        public ICollection<string> ListAllProductName()
        {
            return _db.Products.Select(a => a.ProductName).ToList();
        }

        public ICollection<string> ListAllCategoryName()
        {
            return _db.Categories.Select(a => a.CategoryName).ToList();
        }

        public void Dispose()
        {
            if (_db != null)
                _db.Dispose();
        }
    }
依照先前所說,我們先來做一個讓呼叫端使用的介面吧~我們這邊是以Product資料為例子
public interface IProductRepository
{
     IList<Product> GetProducts();
     Product GetProductByID(int productId);
     Product GetProductByName(string productName);
     void InsertProduct(Product product);
     void DeleteProduct(int productId);
     void UpdateProduct(Product porduct);
}
接著我們要寫一個實作這個介面的類別,並且讓他實際去取出資料。
public class ProductReposity : IRepository
{
     private Northwind _ db = null;
     public ProductReposity(Northwind context)
     {
          if(context == null) throw new ArgumentNullException("context");
          this._db = context;
     }
     public IList<Product> GetProducts()
     {
          return _db.Products.ToList();
     }
     public Product GetProductByID(int productId)
     {
          return _db.Products.Where(a => a.ProductID == productId).Select(a => a).FirstOrDefault();
     }
     public Product GetProductByName(string productName)
     {
          return _db.Products.Where(a => a.ProductName == productName).Select(a => a).FirstOrDefault();
     }
     public void InsertProduct(Product product)
     {
          _db.Products.Add(product);
          _db.SaveChanges();
     }
     public void DeleteProduct(int productId)
     {
          Product product = _db.Products.Find(productId);
          _db.Products.Remove(product);
          _db.SaveChanges();
     }
     public void UpdateProduct(Product product)
     {
          Product target = _db.Products.First(a => a.ProductID == product.ProductID);
          target = product;
          _db.SaveChanges();
     }
以上~在呼叫端時我們就可以這樣使用:
IProductRepository db = new ProductRepository(new Northwind());
var items = db.GetProducts();
foerach(var p in items)
{
     Console.WriteLine("Product Name {0}", p.ProductName);
}
這樣我們就進一步的把呼叫端跟資料庫分離開一點了,不過寫到這邊就打住的話實在會很虛...那我們就把這個情境稍微擴大,變成「有多個資料的操作都是類似的」。依照這樣的寫法我們需要新增一個IRepository類別和與之相對應的Repository類別,既然操作都是類似的能不能介面只寫一份就好了呢?可以的!那我們就來改改吧~
public interface IRepository<T> : IDisposable
{
     IList<T> GetProducts();
     T GetProductByID(int itemId);
     T GetProductByName(string itemName);
     void InsertProduct(T item);
     void DeleteProduct(int itemId);
     void UpdateProduct(T item);
}
而把ProductRepository改成如下:
public class ProductReposity : IRepository<Product>
{
     // 判定是否已經呼叫Dispos()方法。
     private bool disposed = false;

     private Northwind _ db = null;
     public ProductReposity(Northwind context)
     {
          if(context == null) throw new ArgumentNullException("context");
          this._db = context;
     }
     public IList<Product> GetProducts()
     {
          return _db.Products.ToList();
     }
     public Product GetProductByID(int productId)
     {
          return _db.Products.Where(a => a.ProductID == productId).Select(a => a).FirstOrDefault();
     }
     public Product GetProductByName(string productName)
     {
          return _db.Products.Where(a => a.ProductName == productName).Select(a => a).FirstOrDefault();
     }
     public void InsertProduct(Product product)
     {
          _db.Products.Add(product);
          _db.SaveChanges();
     }
     public void DeleteProduct(int productId)
     {
          Product product = _db.Products.Find(productId);
          _db.Products.Remove(product);
          _db.SaveChanges();
     }
     public void UpdateProduct(Product product)
     {
          Product target = _db.Products.First(a => a.ProductID == product.ProductID);
          target = product;
          _db.SaveChanges();
     }
     // 用戶端主動呼叫表明要釋放資源的函式。
     public void Dispose()
     {
         if (_db != null)
             this.Dispose(true);
         //通知GC不用呼叫 finalize 來釋放資源。
         GC.SuppressFinalize(this);
     }
     /// <remarks>
     /// 無法被外部呼叫。
     /// 若是true表示是被用戶端呼叫,manage resource和unmanage resource都可以釋放。
     /// 若是false表示是被GC呼叫,此時應該只釋放unmanage resource。
     /// </remarks>
     protected virtual void Dispose(bool disposing)
     {
         if (!disposed)
         {
             if (disposing)
             {
                 _db.Dispose();
             }
             disposed = true;
         }
     }

     /// <summary>
     /// finalizer。這個無法自行呼叫是由GC來使用的。
     /// </summary>
     ~ProductReposity()
     {
         //以false告知Dispose函數是從垃圾回收器(GC)在呼叫Finalize。
         Dispose(false);
     }
}
ps(這邊的Dispose()寫法的好處是,若呼叫端忘記呼叫Dispose()方法還有Finalize讓GC可以幫忙把物件使用的manage/unmanage resource回收,若呼叫端呼叫Dispose()方法時可以即時把資源進行回收動作,而且不用GC又呼叫一次可以比較有效率點。)
好哩~這樣新的IRepository介面就套用成功了!其實沒什麼變嘛~不過...這樣的話之後若對資料操作的邏輯是相似的話用同一個介面就好了,不需要產生那麼多的介面,也算是省了那麼一點工吧(?) XD

2012年10月5日 星期五

獨立開發Model-使用EF Code First with existing database

前言:
在開發專案時很多時候若有連接到資料庫,通常會把Model這個部分先切割出來開發,這樣可以讓Model獨立在主project之外讓其他projects可以一起使用,在維護的時候其實也會比較方便只需要抽換DLL就好(這也算關注點分離嗎),另外就是Model天生就比較能夠獨立出來開發。不過在獨立開發Model的時候也要注意一些小東西。今天使用Entity Framework (version 5)的Code First with existing database的方式來做個示範。

實作部分:
我們選用的Database是赫赫有名的Northwind資料庫,我想這個資料庫大家應該都比我還熟了。這次我們選出Products和Categories這兩個資料表來做示範的Table吧~!順便提醒一下用Code First with existing database時會有些小地方需要注意!
首先~我們就來把Northwind database bring online吧~


把資料庫online後,接下來就切到Visual Studio中創一個新的Class Library專案吧~
專案名稱就隨便給個Project.Model來代表這個DLL是我們的Model。而因為要用Entity Framework的Code First功能,我們要引用一個EF的DLL:EntityFramework,若是用Nuget來安裝的話會連同System.Data.Entity這個DLL一起裝,不過若是沒有要用視覺設計工具來輔助的話其實是可以不用這個DLL的。
使用Code First的方式來與資料庫做互動的話需要撰寫一個繼承DbContext的類別。這裡我們就把這個類別名稱取為Northwind。

using System.Data.Entity;
///這邊我們只用到Products與Categories這兩個資料表做示範。
///DbSet這個類別代表一組在Context中的實體資料集合,但要注意這個集合內的型別(class)都是相同的,
///不能接受複合型別,像是DbSet<Product,Category>(這邊型別相當於資料庫中的資料表的概念)。
public class Northwind : DbContext
{
    // 這裡的屬性名稱要注意,需要與資料表的名稱相同!
    // 2012-11-16更正!
    // 這邊要注意的應該是DbSet裡的名稱,(也就是Product與Catagory)
    // Case 1:沒有貼Table標籤
    // Code-First預設會依據DbContext中DbSet的泛型名稱(這裡就是Products, Catagories)作為
    // 資料庫中對應的表格名稱。若資料庫中沒有這個表格Code First會幫你產生(很貼心吧,千萬小心)。
    // Case 2:Product有貼Table標籤
    // Code-First會依據標籤所給予的名稱來做為對應到資料庫中的表格名稱。
    public DbSet<Product> Products { get; set; }

    public DbSet<Category> Categories { get; set; }
}
而另外兩個資料表則如下撰寫:
using System.ComponentModel.DataAnnotations.Schema;
public class Product
{
    // Code-First 的預設欄位對應會自動把ID結尾的屬性辨識為PrimaryKey,但可以用[Key]標籤來輔助。
    public int ProductID { get; set; }
    public string ProductName { get; set; }
    public Decimal? UnitPrice { get; set; }
    public bool Discontinued { get; set; }
    /// 注意!使用 Code First 連接既有資料庫沒有加上這個 Schema 標籤會出錯誤!
    /// 因為使用Category與Product資料表,因此需要將這兩張表格的關係寫明。
    /// 若是只引用Product資料表則不用寫也沒關係。
    /// 當然若兩張表格彼此沒有直接關係的話也不用寫這個。
    [ForeignKey("Category")]
    public int CategoryID { get; set; }

    public virtual Category Category { get; set; }
}
另外一張資料表(Category)如下:
using System.Collections.Generic;
public class Category
{
    public int CategoryID { get; set; }
    public string CategoryName { get; set; }
    public string Description { get; set; }
    public byte[] Picture { get; set; }

    public virtual ICollection<product> Products { get; set; }
}
另外我們撰寫另外一個類別來使用Northwind至這個類別(或是說資料庫)~
public class NorthwindRepository : IDisposable
    {
        private Northwind _db;
        //這裡的資料存取為求簡單沒有做依賴注入的動作。
        public NorthwindRepository()
        {
            _db = new Northwind();
        }

        public ICollection<string> ListAllProductName()
        {
            return _db.Products.Select(a => a.ProductName).ToList();
        }

        public ICollection<string> ListAllCategoryName()
        {
            return _db.Categories.Select(a => a.CategoryName).ToList();
        }

        public void Dispose()
        {
            if (_db != null)
                _db.Dispose();
        }
    }

接下來就是要注意的就是使用Code First with existing database時並不會自動幫我們把連線字串準備好,這必須得自行撰寫。因此我們在App.config(這個檔案會在安裝Entity Framework的時候幫我們把基本的東西宣告好)中,需要加入一段<connectionStrings>標籤~
<add name="Northwind"
         connectionString="Data Source=your source;Initial Catalog=Northwind;User Id=uid;Password=pwd" 
         providerName="System.Data.SqlClient"/>
若不確定連線字串如何撰寫可以參考這個網頁,裡面有很多資料庫的連線字串撰寫方法。
以上都寫好了之後其實就可以來使用我們的Northwind資料庫了~這邊我們新創一個空的web form專案,首先我們要先把我們的連線字串放置web.config中,然後就是引用Project.Model.dll了~!不過這邊要注意,引用的dll不能把copy local設為false,否則在create物件時會出現錯誤!
另外使用於console專案時,也可以直接把App.config複製過去,一樣參考DLL時copy local設為true就可以直接拿來用了!像是這樣~
using Project.Model;
static void Main(string[] args)
{
    // 能用using時儘量用,至少沒有害處。
    using (NorthwindRepository db = new NorthwindRepository())
    {
       foreach (var item in db.ListAllProductName())
       {
            Console.WriteLine("item name: {0}", item);
       }
    }
    Console.WriteLine("\nFinish!");
    Console.Read();
}

2012年9月26日 星期三

Using jQuery.ajax in a "Hello World" way

前言:

通常在寫 ASP.NET Web Form 的時候我們很習慣會把一些 html tag 以 Server control 來使用。

但是這些控制項若放在表單中,每次的POST動作都會把這些控制項的內容和狀態一起送回伺服器端,若是有需要的話倒還好,偏偏很多控制項的狀態是根本不需要維護的。像是 Button 啦~ Literal 啦~這些東西其實大多數的時候是死都不會改變的,不過因為天生架構的關係所以每次 POST 回去都會要把這些東西一起送回 Server,實在有點麻煩。所以若是這個時候可以用jQuery來取代 form 傳回 Server 並處理的話可以減輕不少流量的負擔。



實作:

一開始我們先開啟一個空的 Web From 專案~


然後新增一個Index.aspx頁面並設定為起始頁面。

在body中新增下列標籤:

<input type="text" id="textbox1" /><input type="button" id="button1" value="jQuery way" />
    <form id="form1" runat="server">
    <div>
        <asp:TextBox ID="formTextbox1" runat="server"></asp:TextBox>
        <asp:Button ID="formbutton1" runat="server" Text="form way" OnClick="formbutton1_Click" />
    </div>
    </form>


並且在body中新增下列script:

<script type="text/javascript">
    function jqueryAjax() {
        $.ajax({
            url: 'ajaxhandler.ashx',
            type: 'POST',
            dataType: 'text',
            cache: 'false',
            success: function (result) {
                $('#textbox1').val(result);
            }
        })
    };
    $(function () {
        $('#button1').bind('click', jqueryAjax)
    });
</script>


這段script的作用就在於,把id為button1的按鈕的click事件與jqueryAjax函式進行綁定。而jqueryAjax函式的工作就只是呼叫名為ajaxhendler.ashx的泛型處理常式來回應client端的呼叫。

而這次我們在泛型處理常式就直接用最原始的樣貌(就是新增泛型處理常式後直接用),不做任何修改。像是底下這樣:

public void ProcessRequest(HttpContext context)
{
    context.Response.ContentType = "text/plain";
    context.Response.Write("Hello World");
}


是的這次就是個標準的 hello world 級別的程式,不過其實這邊是可以進行資料庫的存取動作的。藉由把資料庫的存取放到這邊來我們可以讓UI的部份動的更順利(ps.若我富堅的毛病沒犯的話,這會在下回分享 = =)

以上這些就算是做好了使用jQuery.ajax方法來與 Server溝通的最簡單方式。底下就是使用jQuery的ajax方法來與Server互動的截圖,附上傳輸量:



另外,我們也要比較一下若是用web form來進行server的溝通的話傳輸量又會是如何:


這邊我們可以發現到,web form就算是簡單的"Hello Word"文字的傳送也需要1.27kb的傳送量,反觀若是用jQuery.ajax來溝通的話只需要344byte的傳輸量,若是傳輸的頁面更複雜的話,兩者的差距會越來越誇張!

所以囉~若是要與Servre溝通的話會建議儘量用ajax來傳輸,減輕網路流量的負擔,但這個範例是用web form,難免會有要用到控制項或是UpdatePanel這種恐怖的東西,若是轉換到ASP.NET MVC架構的話就可以避免和這種怪物正面對決的機會。

不過...若是在做專案的話是要和web form戰鬥還是要跟MVC快樂的舞蹈應該就不是我們工程師能決定的事了....

(ps.以後有機會會分享web form中使用jQuery.ajax的心得...一樣,若我富堅病沒犯的話...)

2012年9月3日 星期一

Single Login in ASP.NET Web Form

前言: 會有這樣的題目主要是因為業主的要求(廢話)。明確的說是這樣的,同一時間同一帳號只能在一個地方登入,之後登入的可以把前面登入的剔除。這個機制其實之前我並沒有想到要怎麼解,或是說沒想到不用資料庫的方式要怎麼解...(超遜的啦!)。不過認真想過後才發現這個機制其實不會太難,或是說我的解法其實很簡單,而且應該也可以用在MVC架構裡。

研究方法:
1)Session。 ASP.NET提供Session讓我們可對工作狀態進行管理,可以讓我們跨越多個要求儲存與瀏覽器工作階段關連的訊息,Session提供Key-value pairs的方式來存取這些資料。雖然我們很常拿Session來儲存使用者的資訊,像是使用者的帳號或是一些其他資訊,不過Session是每個使用者各自有的,這裡面的資訊並不會共享,因此利用Session來達成需求顯然是不行。我們需要的應用系統的全域變數,或是資料結構來對整個系統的使用者進行管理。這樣的需求我想全域應用程式類別(預設檔名為Global.asax)就是我這邊提出的解答!

2)簡述關於Global.asax的運作。
當ASP.NET應用程式收到第一個要求的時候會先使用ApplicationManager建立應用程式定義域(應用程式定義域可以隔離應用程式間的全域變數,而且能夠允許每個應用程式個別進行卸載。)。在所有核心物件初始化後會藉由建立System.Web.HttpApplication類別的實體來執行應用程式。若應用程式中有Global.asax(繼承自HttpApplication)則會改用衍生自HttpApplication的Global.asax來建立執行應用程式的實體。而當處理要求的時候則會執行下述事件:(注意!這邊僅列達成需求出最主要的兩個)

///當要求ASP.NET應用程式中的第一個資源(如:網頁)時呼叫,
///這個方法在應用程式生命週期中只會被呼叫一次。
///我們讓全域變數在這邊進行宣告。
Application_Start(object sender, EventArgs e) { }

///這個方法在MSDN裡面的解釋有點怪,不過在應用程式生命週期中該事件發生也是只有一次,
///就是當IIS重啟或是應用程式正常關閉時就會發起該事件。
///我們讓全域變數所用到的資源在這裡釋放。
Application_End(object sender, EventArgs e) { }
開始撰寫時我們要先準備一下這個管理使用者的容器類別:
/// 系統中記錄目前線上全部使用者的識別號的容器。
public sealed class SystemUserPool
{
    private static ConcurrentDictionary<string, string> _UserPool =
        new ConcurrentDictionary<string, string>();

    /// 記錄系統中新登入的使用者。
    /// userNumber = 登入者的帳號。          
    /// userIp = 登入者的 IP 位址。
    /// 使用者登入成功與否。
    public Boolean AddUser(String userNumber, String userIp)
    {
        return _UserPool.TryAdd(userNumber, userIp);
    }
 
    /// 剔除系統中指定的使用者。
    /// userNumber = 使用者的帳號。
    /// 剔除成功與否。
    public Boolean DeleteUser(String userNumber)
    {
        string str = String.Empty;
        return _UserPool.TryRemove(userNumber, out str);
    }
 
    /// 列出目前系統中在線上的使用者。
    /// 使用者列表。
    public List<string> ListAllUser()
    {
        return _UserPool.Keys.ToList<string>();
    }
 
    /// 判斷使用者是否仍在線上。
    /// userNumber = 欲查明的使用者。
    /// 若在線上則為 true,反之則為 false。
    public Boolean IsOnLine(String userNumber)
    {
        return _UserPool.Keys.Contains(userNumber);
    }
 
    /// 判斷使用者是否從同一個地方登入。
    /// userNumber = 使用者帳號。
    /// userIp = 使用者目前 IP 位址。
    /// 若 userIp 與系統中使用者的 IP 位址一致表示從同一個地方登入。
    public Boolean IsTheSameIP(String userNumber, String userIp)
    {
        String orignalIP = String.Empty;
        if (_UserPool.TryGetValue(userNumber, out orignalIP))
        {
            if (orignalIP == userIp)
                return true;
            return false;
        }
        return false;//若沒能取得,表示使用者已經離線。
    }
 
    /// 更新使用者的 IP 位址。
    /// userNumber = 使用者帳號。
    /// userIP = 使用者新的 IP 位址。
    public void UpdateUserIP(String userNumber, String userIP)
    {
        if (!IsOnLine(userNumber))
            return;
        _UserPool[userNumber] = userIP;
    }
 
    /// 把系統使用者儲蓄池清空。
    public void ClearUserPool()
    {
        _UserPool.Clear();
    }
}


在Application_Start中就這樣寫:
void Application_Start(object sender, EventArgs e) 
{
    Application["UserPool"] = new SystemUserPool();
}
void Application_End(object sender, EventArgs e)
{
    var pool = Application["UserPool"] as SystemUserPool;
    if (pool != null)
        pool.ClearUserPool();
}

以上就是我這次所提出來的解決方案!不過其實最常加在這邊的應該是記錄系統狀態的Log啦~只是Log並不在這次範圍內,以後有機會再說囉~

參考文獻:

1) HttpSessionState

2) ASP.NET應用程式生命週期 IIS 5.0/IIS6.0 IIS7.0

3) ApplicationManager

4) HttpApplication

5) ASP.NET生命週期概觀 --> 強烈推薦寫 ASP.NET Web Form的人可以去看看「網頁生命週期的其他考量」裡面的事件圖

2012年8月30日 星期四

Simple usage of Reflection and Dependency Injection

前言:
這一篇主要是要記錄一下反射(Reflection)的方便用法,作為以後的備忘。底下的範例會順便用一些Depency Injection的觀念來實作。

根據MSDN的描述:
反映 (Reflection) 會提供 Type 型別的物件,用來描述組件、模組和型別。 您可以使用反映來動態建立型別的執行個體、將型別繫結至現有物件,或從現有物件取得型別,並叫用其方法或存取其欄位和屬性。 如果您在程式碼中使用屬性,反映可讓您存取這些屬性。

情境:
在動物園裡面有很多不同的動物園區,每個動物園區都可以看到專屬於該園區的動物。當遊客買票進去時,可以依據票的種類去不同的園區觀賞。假設目前我們有兩個園區:

/// 獅子王動物園區
public class LionKing
{
    private ICollection<string> _Pool;
     
    public LionKink()
    {
         _Pool = new  List<string>{
                          "Simba",
                          "Nala",
                          "Pumbaa",
                          "丁滿"
                      };
    }
    public void SeeAnimals()
    {
         foreach(var name in _pool) 
             Console.WriteLine("在動物園區我看到 {0}."name);
    }
}
/// 憤怒鳥動物園區
public class AngryBird
{
    private ICollection<string> _Pool;

    public AngryBird()
    {
         _Pool = new List<string> {
                         "RedBird",
                         "BlueBird",
                         "YellowBird" 
                     };
     }
     public void SeeAnimals()
     {
         foreach(var name in _pool)
             Console.WriteLine("在動物園區我看到{0}."name);     
     }
}
方法: 通常看到這樣的需求有一種寫法是最簡單直覺的:

class Program
{
     static void Main(string[] args)     
     {
         AngryBird ab = new AngryBird();         
         ab.SeeAnimals();         
         Console.Read();     
     }
}

但你我都知道這樣以後動物園區一多,不但要新增該新增的類別,連呼叫端都要一起改,這樣有點不太方便。有鑑於每個園區提供的服務都差不多,都是讓遊客觀賞的,所以我們需要一個可以把這個服務或是說功能提煉出來的類別,讓這些園區都繼承這個類別,而遊客只需依賴這個類別就好,至於實際要去看哪個園區(或是說實際使用的是哪一個服務)則交由子類別本來決定。依據上述的需求我們可以把SeeAnimals() 這個方法提煉出來,於是有了底下的類別:
/// <summary>
/// 動物園區。
/// </summary> 
public abstract class AnimalZone
{
        /// <summary>
        /// 動物園區的名稱。
        /// </summary>
        public string ZoneName { get; set; }

        /// <summary>
        /// 展示動物園區的所有動物。
        /// </summary>
        /// <returns></returns>
        public abstract ICollection<string> ShowZoneAnimals(); 
}

(ps.我知道這個時候這邊變成吐出一個泛型集合有點怪,但是請繼續看下去吧~)

而我們的兩個園區就可以改寫成這樣:
public class LionKing:AnimalZone
{
     private ICollection<string> _Pool;
     public LionKing()
     {
         this.ZoneName = "獅子王園區";
         _Pool = new List<string> {
                         "Simba",
                         "Nala",
                         "Pombaa",
                         "丁滿",
                         "Scar",
                         "Monky"
                     };
     }

     public override ICollection<string> ShowZoneAnimals()
     {
         return _Pool;
     }
}
public class AngryBird:AnimalZone
{
     private ICollection<String> _Pool;
     public AngryBird()
     {
        this.ZoneName = "憤怒鳥園區";

        _Pool = new List<String>{
                        "RedBird",
                        "BlueBird",
                        "YellowBird",
                        "BlackBird",
                        "GreedBird",
                        "WhiteBird"
                    };
     }

     public override ICollection<string> ShowZoneAnimals()
     {
        return _Pool;
     }
}

好~但是還不夠,讓他們繼承個抽象類別事情還沒有結束,改到目前為止做到一半了。剩下的就是用另外一個類別把這個變化封裝起來,我們讓這個類別依賴 AnimalZone 這個抽象類別,然後由這個類別去執行 AnimalZone 提供的服務,這樣我們以後只需要知道這個類別就可了,或是說服務更動時就去改這裡。

    public class AnimalFactory
    {
        private AnimalZone _animalZone;

        /// <summary>
        /// 建構子,依據傳入的值決定要去哪個園區。
        /// </summary>
        /// <param name="animalzone">動物園區名稱。</param>
        public AnimalFactory(string theAnimalzone)
        {

            if(theAnimalzone=="LionKing")
                this._animalZone = new LionKing();
            else
                this._animalZone = new AngryBird();
        }

        /// <summary>
        /// 看動物去。
        /// </summary>
        public void SeeAnimals()
        {
            foreach (var animal in this._animalZone.ShowZoneAnimals())
            {
                Console.WriteLine("I see {0} in {1} zone.", animal, _animalZone.ZoneName);
            }
        }
    }
之後在用戶端就可以依據傳入的字串來決定可以看到哪些動物,其實若動物園經費不足以後不會擴大(咦?!)的話這樣寫就算結束了。但是事情很多時候不像我們想的這麼美好,若以後動物園園區越來越多,建構子中的判斷式或是 switch case 就會讓程式變得難以維護,若可以把這討厭的判斷消滅該有多好....。這種時候就是 Reflection(反射) 登場當英雄的時候啦!

我們可以藉由反射達成動態的產生實體,不需要 if else 或是 switch case 惱人的判斷了!實際上的改寫方法如下:
這邊我們只需要修改建構子的部份


public AnimalFactory(string theAnimalzone)
{
     var myType = Type.GetType(theAnimalzone);
     var animalzone = Activator.CreateInstance(myType) as AnimalZone;
     this._animalZone = animalzone;
}
我們就只需要這樣短短三行,就可以把未來一卡車的判斷給消滅了!這邊要講一下傳入的字串會像是這樣:YourNameSpace.LionKing
這邊用到主要的類別是 Activator ,用這個可以建立物件型別,而他的 CreateInstance 方法除了可以接受無參數的建構式之外,還支援附帶參數的建構子。藉由這種方式我們在呼叫端只需要輸入相對的型別名稱的字串,就可以直接生出相對應的類別了。呼叫端也只需要知道AnimalFactory 這個類別,而 AnimalFactory 也只依賴 AnimalZone 這個抽象類別。

整個類別的架構圖會長這個樣子:
另外要提醒的是....AnimalFactory 的建構子應該要捕捉例外的...只是我這次偷懶偷很大沒有做,若真的要實際應用的話這是一定要做的。

結論:
這算是一個簡單的設計模式的練習,搭配反射讓應用更為有彈性。這邊我們也可以發現到把變化封裝起來是一個在設計系統時必須考慮到的點,只要可以控制住變化,系統的維護負擔就會相對變小,以後有機會的話會多以設計模式搭配來寫文章。


參考文獻:
System.Activator
http://msdn.microsoft.com/zh-tw/library/system.activator.aspx
 System.Activator.Createinstance
http://msdn.microsoft.com/zh-tw/library/system.activator.createinstance%28v=VS.80%29.aspx

System.Reflection
http://msdn.microsoft.com/zh-tw/library/system.reflection.aspx
Design Pattern: 簡單工廠 (Simple Factory / static Factory)
http://www.javaworld.com.tw/confluence/pages/viewpage.action?pageId=1281

2012年8月22日 星期三

AutoResetEvent的簡單使用

緒論:
這篇文章主要針對System.Threading.AutoResetEvent這個類別的間單使用,自己對執行緒(Thread)也還是幼幼班等級,歡迎看到這篇文章的讀者對於文中有誤或是要補充的不吝賜教。

相關研究: 
System.Threading.AutoResetEvent的定義,根據MSDN: Notifies a waiting thread that an event has occurred. This class cannot be inherited. (向等候的執行緒通知發生事件。這個類別無法被繼承。)
唯一的建構子則為: public AutoResetEvent(bool initialState) 這邊的輸入參數表明了這個事件一開始的信號狀態(signal state)為:1.已收到信號(signaled) 或2.未收到信號(non-signaled)。 這兩者的差異會在實驗方法中稍微看出初始狀態對Thread行為的一些差異。

研究方法:
Step1. 為求環境單純一點,請先開啟一個Console Project(主控台應用程式)。
引用所需要的命名空間,要引用的就是只有這兩個命名空間:
using System;
using System.Threading;

Step2.
class Program
{
    public static Thread t1;
    // 初始狀態為 non-signalled
    public static AutoResetEvent ar1 = new AutoResetEvent(false);
    static void Main(string[] args)
    {
        t1 = new Thread(() => {
             Console.WriteLine("Thread T1 simulating some work for 5 seconds.");
            // 這裡用睡覺模擬執行一些I/O bound的作業。
            Thread.Sleep(5000);
            Console.WriteLine("Thread T1 finished some work.");
            // 設定 ar1 狀態為signaled。讓主執行緒不用等可以繼續跑。
            ar1.Set();
            Console.WriteLine("Thread T1 end.");
            });
        t1.Name = "T1";
        t1.Start();
        ar1.WaitOne();// waite for signal.除非等到狀態為signaled否則主執行緒就會在這裡繼續等下去。
        Console.WriteLine("Main Thread end");
        Console.Read();
    }
}

實驗數據與分析:


當成是一開始執行的時候AutoResetEvent的State就設定為non-signaled,所以主執行緒執行到t1.Start();時產生子執行緒來執行t1的方法(這裡使用lambda statement來偷懶,做一個很間單的事情。)之後,主執行緒往下執行到ar1.WaiteOne();就發現目前State依然是non-signaled,必須等到State設為signaled才能繼續往下執行,因此等到子執行緒中把ar1的State設為Signaled時才會回到主執行緒讓"Main Thread end."最後才列印出來。

ps.你可以試著把ar1的狀態在初始時設為signaled,看看執行起來的結果又是如何?

結論:
當你需要使用多執行緒讓你的I/O Bound作業不影響到整個執行緒的流暢時,可以考慮使用這個簡單的方法讓一些耗費I/O時間的工作移到主執行緒之外,等到需要拿來用之前使用WaiteOne()方法來確保執行完畢後再接續進行後續的工作。

參考文獻:
1.MSDN AutoResetEvent Class
http://msdn.microsoft.com/en-us/library/system.threading.autoresetevent.aspx
2.Code Project Threading系列文章 -- by Sacha Barber   => 特別推薦此系列文章此篇文章參考第三章來實作。

特別附註:
若你用Visual Studio中文版來開發,你會發現IDE對Set()還有Reset()這兩個方法的說明很有問題!但若是英文原版的解釋倒是挺清楚的。唔...若英文不是特別爛的話還是會建議各位直接看原文會比較好,若有問題國外的開發者絕對會馬上開炮,官網也會馬上改。(ps.剛剛已經把訂正過後的翻譯送出審查,不知道什麼時候會改。 讓我生氣的是簡體中文的翻譯竟然沒有問題...那繁體是怎麼回事 = =?!)
有圖有真相:



2012年5月11日 星期五

MediaElement開啟新的Source時的行為

前言:
Silverlight 在多媒體的處理中提供了 MediaElement 這個控制項,可以進行影片的播放、聲音的播放等應用。而串流資訊的來源可以是一個網址列,也可以是一個串流來源。但是若來源固定的話這個應用實在很弱,因此大多的時候來源都是可以隨時變動的。今天在做一些測試的時候有個情境是這樣的:
使用者在 ListBox 中選了一個 Item,而選中後會觸發事件讓 MediaElement 的 Source 重新設定來源,然後就可以直接播放新的 Item 的內容 (不同的 Item 表示不同的 Radio 位址)。

問題描述:
點選 Listbox 的 Item 後會觸發事件,但是程式執行 MediaElement1.Play() 後卻沒有動作。

分析:
根據 MSDN 的官方文件中說明(http://msdn.microsoft.com/en-us/library/cc189079(v=vs.95).aspx),MediaElement 的 Source 在進行變更之後會針對來源進行驗證或是授權(如果有需要的話),接著就會觸發 MediaOpened 事件,不過接著會進行什麼動作則是依據兩個屬性來決定;AutoPlay, CanPause。

Case 1:
若 AutoPlay 是 True。則MediaElement的Source更改後狀態直接進入 Playing 來播放多媒體。

Case 2:
若 AutoPlay 是 False,CanPause 是 True。則Source更改後狀態會進入 Pause。

Case 3:
若兩個屬性都是 False。則 Source 更改後狀態會進入 Stopped。

因此,若想要使用者點選選單然後立刻播放Radio的話可以這麼做:
===================================================
Solution1:
xaml 文件中宣告 MediaElement 與這些屬性
MediaElement Height="200" Name="myMediaRadio" Width="300" AutoPlay="True"
code-behind檔中這樣寫
protected void myRadioList_SelectionChanged(object sender, SelectionChangedEventArgs e)
{
   var radioname = e.AddedItems[0].ToString();
   myMediaRadio.Source = new Uri(_db.GetRadioUri(radioname), UriKind.Absolute);
   myMediaRadio.Volume = 100;
}

========================================================
Solution2:
xaml 文件中 MediaElement 改成這樣:
MediaElement Height="200" Name="myMediaRadio" Width="300" AutoPlay="True" Source="{a uri which can play}"
code-behind 檔要多處理一個 MediaOpened 事件
void myMediaRadio_MediaOpened(object sender, System.Windows.RoutedEventArgs e)
{
   myMediaRadio.Play();//在 MediaOpened 時執行播放,可以避免因上述因素讓媒體進入其他狀態。
}

===========================================================
以上!