模版方法中的方法可以分為兩大類:模版方法(Template Method)和基本方法(Primitive Method)。
a.模版方法
一個模版方法是定義在抽象類中的,把基本操作方法組合在一起形成一個總算法或一個總行為的方法。
這個模版方法一般會在抽象類中定義,并由子類不加以修改地完全繼承下來。
b.基本方法
基本方法又可以分為三種:抽象方法(Abstract Method)、具體方法(Concrete Method)和
鉤子方法(Hook Method)。
抽象方法:一個抽象方法由抽象類聲明,由具體子類實現。在C#語言里一個抽象方法以abstract
關鍵字標示出來。
具體方法:一個具體方法由抽象類聲明并實現,而子類并不實現或置換。在C#語言里面,一個具體方法
沒有abstract關鍵字。
鉤子方法:一個鉤子方法由抽象類聲明并實現,而子類會加以擴展。通常抽象類給出的實現是一個空實現,
作為方法的默認實現。(Visual FoxPro中項目向導建立的項目會使用一個AppHook類實現監(jiān)視項目成員變化,調整系統(tǒng)結構的工作。)鉤子方法的名字通常以do開始。
下面講解窗體基類的模板方法實現:
抽象方法必須定義在抽象類中(Abstract Class),在某些場合不一定使用抽象類,比如frmBase是個具體類,因此我將模板方法改為虛擬(Virtual)的。
??????? ///?
????????///?在基類定義查詢數據的模板方法,由派生窗體實現具體的查詢功能。
????????///?
????????///?
????????protected?virtual?DataTable?DoSearch()
????????{
????????????return?new?DataTable();?//模板方法不能返回實際數據。
????????}
派生類具體實現:
??????? //復寫(Override)模板方法,具體實現
????????protected?override?DataTable?DoSearch()
????????{
????????????return?new?BLL_Product().Search(txtProductCode.Text, txtProductName.Text);
????????}
窗體基類界面:
public?partial?class?frmBase : Form
{
???public?frmBase()
???{
??????InitializeComponent();
???}
???
???private?void?btnSearch_Click(object?sender, EventArgs e)
???{
??????//調用模板方法?
??????DataTable data =?this.DoSearch();
??????
??????if?(data.Rows.Count > 0)
??????this.dataGridView1.DataSource = data;
??????else
??????MessageBox.Show("沒有找到數據!");
???}
???
???///??
???///??
???protected?virtual?DataTable DoSearch()
???{
??????return?new?DataTable();?//模板方法不能返回實際數據。?
???}
???
}
派生一個產品定義窗體:
public?partial?class?frmProduct : CSFramework.TemplateMethodDemo.frmBase
{
???public?frmProduct()
???{
??????InitializeComponent();
???}
???
???//復寫(Override)模塊方法,具體實現?
???protected?override?DataTable DoSearch()
???{
??????return?new?BLL_Product().Search(txtProductCode.Text, txtProductName.Text);
???}
}
搜索功能在窗體基類實現了,我們只需要定義具體的數據查找方法。
通過重寫DoSearch模塊方法,從業(yè)務邏輯層返回數據。
///?
????///?業(yè)務邏輯層
????///?
????public?class?BLL_Product
????{
????????public?DataTable?Search(string?productCode,?string?productName)
????????{
????????????string?where =?" 1=1 ";
?
????????????if?(!String.IsNullOrEmpty(productCode))
????????????????where = where +?" and ProductCode=’"?+ productCode +?"’";
?
????????????if?(!String.IsNullOrEmpty(productName))?//支持模糊查找
????????????????where = where +?" and ProductName like ’%"?+ productName +?"%’";
?
????????????string?sql =?"select * from tb_Product where "?+ where;
????????????//從SQL Server?獲取數據?..........省略...........
????????????//
????????????//下面返回測試數據
????????????DataTable?test =?this.GetDemoData();//返回測試數據
????????????DataRow[] rows = test.Select(where);
?
????????????//克隆?System.Data.DataTable?的結構,包括所有?System.Data.DataTable?架構和約束。
????????????DataTable?retTable = test.Clone();
????????????foreach?(DataRow?row?in?rows)
????????????????retTable.ImportRow(row);
?
????????????return?retTable;
????????}
?
????????///?
????????///?構建測試數據表
????????///?
????????///?
????????private?DataTable?GetDemoData()
????????{
????????????DataTable?dt =?new?DataTable();
????????????dt.Columns.Add("ProductCode",?typeof(String));
????????????dt.Columns.Add("ProductName",?typeof(String));
????????????dt.Columns.Add("Price",?typeof(Decimal));
?
????????????dt.Rows.Add("001",?"原創(chuàng)作品CD", 99);
????????????dt.Rows.Add("002",?"Dev Express 9.2x Suit", 9960);
????????????dt.Rows.Add("003",?"可口可樂", 1.90);
????????????dt.Rows.Add("004",?"Engima CD", 269);
?
????????????return?dt;
????????}
????}