嘗試.NET 4.5中狀態(tài)機(jī)工作流
NET 4.5 中狀態(tài)機(jī)工作流
結(jié)尾的總結(jié)有點(diǎn)與本文主題不符,是寫文章時(shí)想到的,就順手寫上了(^_^)
例子下載:
http://files.cnblogs.com/foundation/Wxwinter.Test.Statements.rar
介紹
.NET 4.5 中WF已為原WF4.0提供了狀態(tài)機(jī)模式的支持.
先看一下增加加的Activity
使用VS2011創(chuàng)建WorkFlow項(xiàng)目,選 .NET 4.5 的WF模板
在WF設(shè)計(jì)界面可以看到狀態(tài)機(jī)模板
由3個(gè)Activity組成
StateMachin 是狀態(tài)機(jī)容器
State 是狀態(tài)節(jié)點(diǎn)
FinalState 是結(jié)束狀態(tài)節(jié)點(diǎn)
在State 中具有[進(jìn)入狀態(tài)],[結(jié)束狀態(tài)]兩個(gè)執(zhí)行區(qū),可以添加任意Activity,
FinalState 中具有[進(jìn)入狀態(tài)]
State可以添加多條分支,在分支上可以設(shè)置[Trigger],[Condition],[Action],這些與WF4.0中其他條件類Activity設(shè)置一樣
例子
用一個(gè)例子演示一下狀態(tài)機(jī)的使用
流程業(yè)務(wù)模型
設(shè)計(jì)書簽
- publicsealedclassresultBookmark: NativeActivity
- {
- publicInArgumentbookmarkName { getset}
- protectedoverrideboolCanInduceIdle
- {
- get
- { returntrue}
- }
- protectedoverridevoidExecute(NativeActivityContextcontext)
- {
- stringbookmark = context.GetValue(bookmarkName);
- context.CreateBookmark(bookmark, newBookmarkCallback(bookmarkCallback));
- System.Console.WriteLine("創(chuàng)建bookmark:{0}", bookmark);
- }
- voidbookmarkCallback(NativeActivityContextcontext, Bookmarkbookmark, objectobj)
- {
- if(obj != null)
- {
- this.Result.Set(context, obj.ToString());
- }
- else
- {
- this.Result.Set(context, string.Empty);
- }
- }
- }
在設(shè)計(jì)界面用WF構(gòu)建業(yè)務(wù)模型
設(shè)計(jì)State,以[A]節(jié)點(diǎn)為例,其他類似
#p#
設(shè)計(jì)宿主
- voidworkflowCompleted(WorkflowApplicationCompletedEventArgse)
- {
- instance = null
- System.Console.WriteLine("workflowCompleted:{0}", e.CompletionState.ToString());
- }
- voidaborted(WorkflowApplicationAbortedEventArgse)
- {
- instance = null
- System.Console.WriteLine("aborted ,Reason:{0}", e.Reason.Message);
- }
- UnhandledExceptionActionunhandledExceptionl(WorkflowApplicationUnhandledExceptionEventArgse)
- {
- System.Console.WriteLine("unhandledException:{0}", e.UnhandledException.Message);
- returnUnhandledExceptionAction.Cancel;
- }
- voidworkflowIdel(WorkflowApplicationIdleEventArgse)
- {
- System.Console.WriteLine("Idle:{0}", e.InstanceId);
- System.Console.WriteLine("--------BookmarkName---------------------------");
- foreach(varitem ine.Bookmarks)
- {
- System.Console.WriteLine("{0}", item.BookmarkName);
- }
- System.Console.WriteLine("================================");
- }
- WorkflowApplicationinstance = null
- privatevoidstartButton_Click(objectsender, EventArgse)
- {
- valueComboBox.Items.Clear();
- valueComboBox.Items.Add("A");
- valueComboBox.Items.Add("B");
- valueComboBox.Items.Add("C");
- valueComboBox.Items.Add("E");
- valueComboBox.SelectedItem = "B"
- instance = newWorkflowApplication(newmyWorkflow());
- instance.Completed = newAction(workflowCompleted);
- instance.OnUnhandledException = unhandledExceptionl;
- instance.Aborted = aborted;
- instance.Idle = workflowIdel;
- instance.Run();
- }
- privatevoidsubmitButton_Click(objectsender, EventArgse)
- {
- stringbookName =bookmarkTextBox.Text;
- stringinputValue = valueComboBox.SelectedItem.ToString();
- if(instance != null)
- {
- if(instance.GetBookmarks().Count(p => p.BookmarkName == bookName) == 1)
- {
- instance.ResumeBookmark(bookName, inputValue);
- }
- else
- {
- foreach(varv ininstance.GetBookmarks())
- {
- System.Console.WriteLine("--------請(qǐng)從下面選項(xiàng)中選擇一個(gè)BookmarkName---------------------------");
- System.Console.WriteLine("BookmarkName:{0}:,OwnerDisplayName:{1}", v.BookmarkName, v.OwnerDisplayName);
- System.Console.WriteLine("================================");
- }
- }
- }
- else
- {
- MessageBox.Show("沒有創(chuàng)建實(shí)例");
- }
- }
測(cè)試
總結(jié)
這本是一個(gè)應(yīng)在NET 4.0 中實(shí)現(xiàn)的功能,其實(shí)不談數(shù)據(jù)結(jié)構(gòu)與運(yùn)行機(jī)制,FlowChar完全可以實(shí)現(xiàn)StateMachin 的所有業(yè)務(wù)邏輯.
只不過StateMachin 在VS設(shè)計(jì)器中給人一種更整潔的感覺,不過這只是展現(xiàn)層面的東西,之前不少人都實(shí)現(xiàn)過這種效果(包括我自已),理論上并不是很難,但由于WF4.0大部分的功能類從繼承Object后就開始封閉與工程級(jí)保護(hù),在這種情況下別說繼承,就正常連引用都辦不到,當(dāng)然也有辦法解決,不過實(shí)現(xiàn)后一般只做為概念展現(xiàn),所以很多人在實(shí)際使用時(shí)采取的是自定義流程設(shè)計(jì)器,將FlowChar的結(jié)構(gòu)展現(xiàn)成StateMachin ,通過攔截Bookmark實(shí)現(xiàn)[進(jìn)入狀態(tài)],[結(jié)束狀態(tài)],其實(shí)這種工作量比傳統(tǒng)意義上開發(fā)一套業(yè)務(wù)流程平臺(tái)還有大,但為什么還有使用WF,這是很多人不理解的.其實(shí)使用WF的人基本分為兩類,一類是具備設(shè)計(jì)器開發(fā)能力的人,看到VS中的WF設(shè)計(jì)界面后就覺得WF為其提供了一套拿來就能用的業(yè)務(wù)流程平臺(tái).這種想法是不對(duì)的,暫不分析WF是否只接適用用戶業(yè)務(wù),就常識(shí)來說,與WCF,WPF并列的WF怎么可能是直接面對(duì)業(yè)務(wù)用戶的.另一類選擇WF的人是看中了WF對(duì)狀態(tài)的副本處理,事物處理,補(bǔ)償處理這些從COM+延續(xù)下來的理念.很多時(shí)候我一直認(rèn)為WCF +WF就是一個(gè)新版本的COM+
最后不得不提一下,雖然WF實(shí)現(xiàn)了FlowChar與StateMachin ,但其數(shù)據(jù)結(jié)構(gòu)并不是圖,而是樹.如果既對(duì)Expression Tree熟悉,又對(duì)WF熟悉,你會(huì)發(fā)現(xiàn)Expression Tree就是一個(gè)WF數(shù)據(jù)結(jié)構(gòu)精簡(jiǎn)版.談到Expression Tree了解的人是多了,我也好解釋了,這種數(shù)結(jié)構(gòu)其實(shí)與內(nèi)存中的棧管理方式很象,這就決定了調(diào)用只能指向棧開始,而不能任意指向棧的任意位置(不談溢出,GOTO指針這個(gè)層面的問題).
為什么提上面這段內(nèi)容,是因?yàn)槲乙f明由于這種數(shù)據(jù)結(jié)構(gòu)我們很難現(xiàn)以下功能(這是一個(gè)業(yè)務(wù)功能,我先用C#風(fēng)格有代碼描述一下):
上面這段代碼看起來有些瘋狂,我們不從代碼層面考慮(不加條件判斷,不提動(dòng)態(tài)編譯,ICO,AOP),我只想這樣寫,我希望當(dāng)程序運(yùn)行到[24]時(shí)自動(dòng)告訴我缺少哪個(gè)上下文(如obj),并讓我補(bǔ)齊上下文后繼續(xù)執(zhí)行.
將這個(gè)問題業(yè)務(wù)化:
上車前要買票,(買票是上車的前置條件)
但我已經(jīng)上了車卻沒買票,是補(bǔ)票還是將我趕下車買完票再上車.這就是經(jīng)典的[補(bǔ)償處理]機(jī)制
程序員說:"你們的業(yè)務(wù)總變,你們都說不清,讓我們?cè)趺撮_發(fā)"
客戶說:"我們要是業(yè)務(wù)從來不變,每個(gè)人都能說清,要你們開發(fā)系統(tǒng)做什么"
數(shù)學(xué)是嚴(yán)謹(jǐn)?shù)?而真實(shí)的業(yè)務(wù)到處充滿了變通.如何用嚴(yán)謹(jǐn)?shù)臄?shù)學(xué)來構(gòu)建變通業(yè)務(wù)的邏輯模型才是從事企業(yè)應(yīng)用開發(fā)程序員所面臨的問題
原文鏈接:http://www.cnblogs.com/foundation/archive/2011/09/19/2181324.html
【編輯推薦】
- 調(diào)用ASP.NET工作流:承載及限制
- ASP.NET工作流的創(chuàng)建與啟動(dòng)
- ASP.NET數(shù)據(jù)庫(kù)連接類ClassConn以及oledb數(shù)據(jù)庫(kù)連接方法
- ASP.NET數(shù)據(jù)庫(kù)驅(qū)動(dòng)類的實(shí)現(xiàn):DBHelper
- 用C#寫的ASP.NET數(shù)據(jù)庫(kù)操作類