自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

.NET程序員應(yīng)該熟悉的開(kāi)發(fā)模式

開(kāi)發(fā) 后端
下面是.NET程序員應(yīng)該熟悉的幾種開(kāi)發(fā)模式,當(dāng)然也就包括了MVC模式等等。希望對(duì)大家有所幫助。

我們總會(huì)有這樣一個(gè)經(jīng)驗(yàn):一個(gè)系統(tǒng)最不容易也最不應(yīng)該變化的部分是領(lǐng)域邏輯,最容易變化也最應(yīng)該變化的是數(shù)據(jù)的呈現(xiàn)方式。

在Java的各種應(yīng)用中可以說(shuō)是到處可見(jiàn)MVC,J2EE貫穿MVC的概念,android的開(kāi)發(fā)方式也是類MVC的,MVC結(jié)構(gòu)對(duì)于做過(guò)Java應(yīng)用的人而言簡(jiǎn)直就是司空見(jiàn)慣。而在.NET這邊,由于之前微軟為大家提供的各種winform、ASP.NET項(xiàng)目典范(比如那個(gè)petshop series)將“三層”概念很好的灌輸?shù)搅?NET程序員的大腦中,許多.NET開(kāi)發(fā)者凡是做個(gè)東西都要搬出自己最拿手的IModel、IDAL這樣的神器。

其實(shí)MVC與所謂的“三層架構(gòu)”是兩個(gè)層次上的東西,前者是一種結(jié)構(gòu)模式,而后者則是分層的角度去說(shuō)。

一件很奇怪的事情,許多人知道“三層”卻不知道MVC,其實(shí)這要?dú)w結(jié)與.NET的早期開(kāi)發(fā)技術(shù)ASP.NET和winform這些page controller的典范讓許多人對(duì)三層夸夸其談卻對(duì)MVC視而不見(jiàn)甚至一無(wú)所知。什么是page controller模式呢?搞.NET的大多都用過(guò)winform和webform,這種xxxform用起來(lái)很直觀,我們想要做一個(gè)程序,ok,最簡(jiǎn)單的方式就是拖拖拽拽幾個(gè)控件,然后在一個(gè)叫code behind的東西里寫(xiě)這些UI事件的處理邏輯,加一大堆變量用于記錄數(shù)據(jù)和狀態(tài),這樣一個(gè)程序就能出爐。這種開(kāi)發(fā)方式對(duì)于一些小軟件系統(tǒng)的開(kāi)發(fā)其實(shí)效率還是蠻高的,后來(lái)人們看到其弊端---一旦修改UI,事件處理就要跟著變,但是業(yè)務(wù)還是那個(gè)業(yè)務(wù),憑什么要修改非UI的代碼?于是有人提出“三層”,最樸素的理解就是將原本那堆事件處理里的code分成業(yè)務(wù)代碼和數(shù)據(jù)庫(kù)訪問(wèn)代碼并轉(zhuǎn)移到其它類中,做多了就把那坨UI叫做UI,那坨業(yè)務(wù)代碼叫做BLL,那坨DAO叫做DAL。也就是這種架構(gòu):

image

而對(duì)于J2EE的開(kāi)發(fā)者來(lái)說(shuō)熟悉的是下圖。 

image

(說(shuō)明:這兩幅圖copy自是daxnet文)

MVC是什么

MVC是一個(gè)很經(jīng)典的結(jié)構(gòu),并且其又其思想衍生出很多變種比如MVP,MVVP。傳統(tǒng)的MVC結(jié)構(gòu)之一是這樣的(拿主動(dòng)型MVC來(lái)說(shuō)):

image

比如web開(kāi)發(fā)(比如ASP.NET MVC或者是Java的web開(kāi)發(fā)方式),view就是純web頁(yè)面或者webservice,當(dāng)提交一個(gè)表單/調(diào)用webservice或者ajax后會(huì)將數(shù)據(jù)提交給controller(當(dāng)然期間可能會(huì)經(jīng)過(guò)各種filterchain、listener這樣的東西)controller調(diào)用相應(yīng)的業(yè)務(wù)模塊來(lái)處理這個(gè)請(qǐng)求,最終結(jié)果會(huì)更新View的顯示。

MVP

對(duì)于非天然MVC的框架

對(duì)于ASP.NET/winform而言,雖然可以通過(guò)改造讓其支持MVC結(jié)構(gòu)的開(kāi)發(fā)(比如通過(guò)定制IHttpModule、IHttpHandler云云),但是在企業(yè)看來(lái)這些都算是邪門(mén)武功(因?yàn)檫@樣會(huì)喪失xxxform在開(kāi)發(fā)上的很多特性比如快速開(kāi)發(fā))。大多數(shù)使用的是mvp模式。什么是mvp呢?其實(shí)mvp是MVC的一個(gè)變種。因?yàn)橛脀inform或者webform的話form始終是個(gè)阻礙MVC開(kāi)發(fā)的問(wèn)題。那么好,我們?nèi)匀皇褂胐esigner和codebehind,其實(shí)一個(gè)架構(gòu)設(shè)計(jì)的好壞是取決于人而不是具體的技術(shù)的,只要我們OO一時(shí)強(qiáng)page controller一樣好用。

image

在MVP模式中我們需要自己定制各個(gè)View(web頁(yè)面或者窗體)對(duì)應(yīng)的IView和IPresenter、IModel。IView要對(duì)IPresenter暴露操作UI、數(shù)據(jù)綁定的接口,IPresenter對(duì)IView要暴露當(dāng)UI事件觸發(fā)需要調(diào)用的接口,IPresenter根據(jù)IView傳遞過(guò)來(lái)的請(qǐng)求調(diào)用業(yè)務(wù)接口并根據(jù)結(jié)果操作UI。舉個(gè)簡(jiǎn)單的例子,一個(gè)計(jì)算“x+y=?”的程序。如果我們這樣定義IPresenter和IView

  1. public interface IPresenter   
  2.     {   
  3.         IView View { getset; }   
  4.         void CalculateResult();   
  5.     }  
  6.  
  7. public interface IView   
  8.     {   
  9.         IPresenter Presenter { getset; }   
  10.         void ShowResult(string result);   
  11.         int ValueOne { get; }   
  12.         int ValueTwo { get; }   
  13.     } 

IPresenter的實(shí)現(xiàn)如下(這里從簡(jiǎn)把IModel去掉了)

Presenter

  1. namespace ClientLibrary  
  2. {  
  3.     public class Presenter : IPresenter  
  4.     {  
  5.         private IView _view;  
  6.         public IView View  
  7.         {  
  8.             get 
  9.             {  
  10.                 return _view;  
  11.             }  
  12.             set 
  13.             {  
  14.                 _view = value;  
  15.                 _view.Presenter = this;  
  16.             }  
  17.         }  
  18.  
  19.         private static readonly string RESULT_FORMATTER = "{0}+{1},the result is {2}";  
  20.         public void CalculateResult()  
  21.         {  
  22.             if (_view != null)  
  23.             {  
  24. var result = string.Format(RESULT_FORMATTER, _view.ValueOne, _view.ValueTwo, _view.ValueOne + _view.ValueTwo);  
  25.                 _view.ShowResult(result);  
  26.                 this.A = 123;  
  27.             }  
  28.         }  
  29.         private int _a;  
  30.         public int A  
  31.         {  
  32.             set 
  33.             {  
  34.                 A = value;  
  35.             }  
  36.         }  
  37.     }  
View的實(shí)現(xiàn)如下(那silverlight為例,換成別的也行)

MainPage

  1. namespace debug  
  2. {  
  3.     public partial class MainPage : UserControl, IView  
  4.     {  
  5.         public MainPage()  
  6.         {  
  7.             InitializeComponent();  
  8.         }  
  9.  
  10.         private IPresenter _presenter;  
  11.  
  12. private void btn_Click(object sender, RoutedEventArgs e)  
  13.         {  
  14.             if (_presenter != null)  
  15.             {  
  16.                 _presenter.CalculateResult();  
  17.             }  
  18.             #region hidden  
  19.             /*int total = 0;  
  20.             try  
  21.             {  
  22. total = int.Parse(tb1.Text) + int.Parse(tb2.Text);  
  23. MessageBox.Show("計(jì)算結(jié)果:" + total.ToString());  
  24.             }  
  25.             catch (Exception ex)  
  26.             {  
  27.                 MessageBox.Show("出錯(cuò)啦" + ex.ToString());  
  28.             }  
  29.             finally  
  30.             {  
  31.                 tb1.Text = string.Empty;  
  32.                 tb2.Text = string.Empty;  
  33.             }*/ 
  34.             #endregion  
  35.  
  36.         }  
  37.  
  38.         public IPresenter Presenter  
  39.         {  
  40.             get 
  41.             {  
  42.                 return _presenter;  
  43.             }  
  44.             set 
  45.             {  
  46.                 _presenter = value;  
  47.             }  
  48.         }  
  49.  
  50.         public void ShowResult(string result)  
  51.         {  
  52.             MessageBox.Show(result);  
  53.         }  
  54.  
  55.         public int ValueOne  
  56.         {  
  57.             get { return int.Parse(tb1.Text); }  
  58.         }  
  59.  
  60.         public int ValueTwo  
  61.         {  
  62.             get { return int.Parse(tb2.Text); }  
  63.         }  
  64.     }  

一個(gè)很簡(jiǎn)單的東西,看上去寫(xiě)成的要多些那么一坨東西,但是好處是顯而易見(jiàn)的,就是更換view非常方便,根本不用去改你的IPresenter、Presenter和業(yè)務(wù)。一切都是接口調(diào)用而不依賴具體實(shí)現(xiàn),這就是好處。

你必須要懂的MVVM

對(duì)于.NET平臺(tái)的開(kāi)發(fā)人員,托微軟的福分我們擁有一種更為強(qiáng)大的模型---MVVM。這應(yīng)該算是做WPF/Silverlight應(yīng)用的人必懂的一種結(jié)構(gòu),WPF/silverlight天生支持?jǐn)?shù)據(jù)綁定和命令綁定(不過(guò)sl在命令綁定上還比較弱),這就為我們使用MVVM創(chuàng)造了可能。

View是什么呢,純的View只有xaml或者附帶必要的只與View本身相關(guān)邏輯代碼。ViewModel,你可以把它理解為View具體呈現(xiàn)內(nèi)容所依賴數(shù)據(jù)的一個(gè)抽象,在MVVM中View與ViewModel總會(huì)有一種綁定關(guān)系,一旦ViewModel中被綁定的數(shù)據(jù)發(fā)生改變View上的數(shù)據(jù)就會(huì)跟著變,相反也有可能,比如你的賬號(hào)密碼框內(nèi)容發(fā)生變化,關(guān)聯(lián)的ViewModel中的數(shù)據(jù)就會(huì)被框架自動(dòng)通知到。

在wpf/silverlight中,綁定是通過(guò)xaml語(yǔ)法來(lái)完成(雖然你可以選擇用c#來(lái)寫(xiě)但不符合mvvm的宗旨),并且綁定雙方的通知機(jī)制是有框架來(lái)完成,也就是說(shuō)一個(gè)會(huì)xaml和blend的美工只需事先和coder商量下“咱們的xx和xx是在哪個(gè)ViewModel上叫XXX的屬性的XXX屬性……”問(wèn)題之后就可以各干各的了。那么ViewModel怎么寫(xiě),咋view中又怎么綁定到viewmodel呢?首先我們談ViewModel。

說(shuō)道ViewModel你需要知道依賴屬性和依賴對(duì)象的概念,這是wpf/silverlight的基礎(chǔ)所以不多說(shuō)。有兩種方式寫(xiě)ViewModel。***種是自己去實(shí)現(xiàn)INotifyPropertyChanged接口,并在屬性變化時(shí)去調(diào)用NotifyPropertyChanged事件。

為了方便我們定義一個(gè)ViewModelBase的抽象基類,然后讓其他ViewModel繼承這個(gè)基類。

ViewModelBase

  1. public abstract class ViewModelBase : System.ComponentModel.INotifyPropertyChanged, IDisposable   
  2.     {   
  3.         public event System.ComponentModel.PropertyChangedEventHandler PropertyChanged;   
  4.         protected void OnPropertyChanged(string propertyName)   
  5.         {   
  6.             if (PropertyChanged != null)   
  7.             {   
  8.       var arg = new System.ComponentModel.PropertyChangedEventArgs(propertyName);   
  9.                 PropertyChanged(this, arg);   
  10.             }   
  11.         }   
  12.         public virtual void Dispose()   
  13.         {   
  14.              
  15.         }   
  16.     } 

 

  1. DemoViewModel public class DemoViewModel : ViewModelBase     
  2.  {          
  3. #region fields          
  4. private string _propertyA;          
  5. #endregion          
  6. #region presentation properties          
  7. public string PropertyA          
  8. {              
  9. get              
  10. {                 
  11.  return _propertyA;           
  12.    }              
  13. set        
  14.       {                  
  15. if (_propertyA != value)                
  16. {                      
  17. _propertyA = value;                   
  18. base.OnPropertyChanged("PropertyA");             
  19.      }         
  20.      }        }        
  21.   #endregion    } 

第二種是利用DependencyObject和DependencyProperty。

PeopleItemViewModel

  1. public class PeopleItemViewModel : DependencyObject, IPeopleItemViewModel  
  2.     {  
  3.         public PeopleItemViewModel()  
  4.         {  
  5.               
  6.         }  
  7. public static readonly DependencyProperty SimpleUserDataProperty = DependencyProperty.Register("SimpleUserData"typeof(SimpleUserData), typeof(PeopleItemViewModel));  
  8. public static readonly DependencyProperty RelativeSimpleUserDataProperty = DependencyProperty.Register("RelativeSimpleUserData"typeof(ObservableCollection<SimpleUserData>), typeof(PeopleItemViewModel));  
  9. public static readonly DependencyProperty AllSimpleUserDataProperty = DependencyProperty.Register("AllSimpleUserData"typeof(ObservableCollection<SimpleUserData>), typeof(PeopleItemViewModel));  
  10.  
  11.         public SimpleUserData SimpleUserData  
  12.         {  
  13.             get 
  14.             {  
  15.                 return (SimpleUserData)base.GetValue(SimpleUserDataProperty);  
  16.             }  
  17.             set 
  18.             {  
  19.                 if (!base.CheckAccess())  
  20.                 {  
  21.                     Dispatcher.Invoke(new Action(  
  22.                         () =>  
  23.                         {  
  24.                             SimpleUserData = value;  
  25.                         }));  
  26.                 }  
  27.                 else 
  28.                     base.SetValue(SimpleUserDataProperty, value);  
  29.             }  
  30.         }  
  31.         public ObservableCollection<SimpleUserData> RelativeSimpleUserData  
  32.         {  
  33.             get 
  34.             {  
  35. return (ObservableCollection<SimpleUserData>)base.GetValue(RelativeSimpleUserDataProperty);  
  36.             }  
  37.             set 
  38.             {  
  39.                 if (!base.CheckAccess())  
  40.                 {  
  41.                     Dispatcher.Invoke(new Action(  
  42.                         () =>  
  43.                         {  
  44.                             RelativeSimpleUserData = value;  
  45.                         }));  
  46.                 }  
  47.                 else 
  48.                 {  
  49. base.SetValue(RelativeSimpleUserDataProperty, value);  
  50. var collectionView = CollectionViewSource.GetDefaultView(value);  
  51. collectionView.SortDescriptions.Add(new SortDescription("Distance", ListSortDirection.Ascending));  
  52.                 }  
  53.             }  
  54.         }  
  55.         public ObservableCollection<SimpleUserData> AllSimpleUserData  
  56.         {  
  57.             get 
  58.             {  
  59.       return (ObservableCollection<SimpleUserData>)base.GetValue(AllSimpleUserDataProperty);  
  60.             }  
  61.             set 
  62.             {  
  63.                 if (!base.CheckAccess())  
  64.                 {  
  65.                     Dispatcher.Invoke(new Action(  
  66.                         () =>  
  67.                         {  
  68.                             AllSimpleUserData = value;  
  69.                         }));  
  70.                 }  
  71.                 else 
  72.                 {  
  73. base.SetValue(AllSimpleUserDataProperty, value);  
  74. var collectionView = CollectionViewSource.GetDefaultView(value);  
  75. collectionView.SortDescriptions.Add(new SortDescription("Distance", ListSortDirection.Ascending));  
  76.                 }  
  77.             }  
  78.         }  

在View中綁定ViewModel。

為了方便,我們可以在app.xaml中將需要的viewmode放到全局資源字典中。

image

然后再我們的vs視圖設(shè)計(jì)器Properties(中文版顯示的是“屬性”)頁(yè)上選擇為綁定源設(shè)置綁定目標(biāo)(包括source和path等)以及必要的值轉(zhuǎn)換器等等即可。

image image image

(PS:雖然vs很強(qiáng)大,但個(gè)人還是建議熟悉xaml的綁定語(yǔ)法,想當(dāng)初用vs2008搞wpf的時(shí)候貌似還沒(méi)有這么方便的設(shè)計(jì)器。。。)

原文鏈接:http://www.cnblogs.com/wJiang/archive/2010/12/11/1903039.html

【編輯推薦】

  1. .NET Framework字符串相關(guān)操作細(xì)節(jié)介紹
  2. 詳解.NET字符串解析的具體過(guò)程
  3. 改進(jìn)C#連接字符串的性能
  4. .NET Lambda表達(dá)式的語(yǔ)義:字符串列表范例
  5. C#字符串的幾種常用方法

 

責(zé)任編輯:彭凡 來(lái)源: 博客園
相關(guān)推薦

2011-03-10 13:45:24

VB.NET

2014-11-10 09:46:57

程序員

2009-03-13 15:18:45

程序員飲食雜談

2017-12-19 20:35:22

程序員中興事件自殺

2016-04-11 17:49:33

程序員外包

2013-04-01 15:51:09

程序員管理

2015-08-11 09:20:51

初級(jí)程序員Linux命令

2021-03-02 15:31:37

程序員技能開(kāi)發(fā)者

2009-06-19 14:38:20

Java程序員

2014-08-01 10:18:16

.Netdump

2014-07-16 09:34:44

2020-12-08 13:23:34

程序員熬夜加班

2012-11-30 11:29:05

高級(jí)程序員程序員

2009-02-13 09:45:27

程序員JavaPHP

2010-10-29 09:08:57

PHPMySQL

2012-04-19 12:50:51

Java

2015-08-20 09:06:48

程序員

2016-01-18 11:03:58

程序員搜索技巧

2013-08-20 09:33:59

程序員

2013-12-02 10:10:35

Python工具庫(kù)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)