內容提要:
如果想試驗一下自己對一個知識到底有幾分的掌握,最好的方法就是實踐她。結合一個名為“NOTE”的項目(記事本),運用MVC模式進行構建,一起體會一下MVC的強大與靈活。請注意的是,本文中關于不再使用MVC這樣的術語,而是使用邊界類、控制類、實體類、生命周期類這樣的稱呼。關于MVC的在j2me上的種種變化請看本文的(一),尤其看看我的習慣做法,否則交流上有困難。
本文重視的是總體設計,細節上沒有過多的談。
I want a good sleep.
版權聲明:
本文同時發表在www.j2medev.com和我的Blog(blog.csdn.net/alikeboy)上,如果需要轉載,有三個途徑:1)聯系我并經我同意;2)和www.j2medev.com有轉載文章合作協議的 3)通過Rss聚合我的Blog。另外網上轉載需要全文轉發(包括文章的頭部的聲明),不要斷章取義。
正文:
Note項目描述手機是一個真正隨身攜帶的數字終端,我們除了利用手機打電話、發信息外,往往讓他幫助我們記錄文字性的信息。PDA有較大的屏幕和手寫輸入的功能,而手機上的記事功能有限,記錄的內容很簡單??梢哉f利用手機記錄信息是一種被視為理所應當的功能,也造就了一個一系列的應用——將手機作為隨時隨地的信息收集器。不難設想此類應用會用一定的市場,我們今天選擇的Note記事本項目,正是此類應用的單機版的原形。
Note是很多手機的內建應用程序,一般叫做記事本或便簽,說明這是一個非常常用的服務。也許朋友們可能認為開發一個已有的程序沒有挑戰性。是的,在開發上最忌諱的就是重作車輪,但對于教學則可兩說著。畢竟一個簡單的原型程序不會讓我們陷入太多的細節,我的主旨是向大家介紹MVC模式的應用方法。
Note為手機用戶提供記錄一些簡短信息的功能,用戶可以添加記錄,打開瀏覽記錄,并可以隨時修改已經保存在手機上的記錄,當然也可以刪除它們。就好象Windows下的記事本一樣,只不過多了管理的功能。
用例分析首先設想一下誰在使用這個程序:手機用戶。好,我們以后就稱這個參與者為用戶(user)。
然后設想一下,用戶都利用我們的NOTE(中文我們叫做記錄好了)干些什么呢?很顯然,用戶可以添加新的記錄,瀏覽他添加的記錄,修改他所添加的記錄,并且他還可以刪除記錄。一條記錄應該簡單的包括用戶對記錄起的名字,記錄的創建或修改時間,以及最重要的記錄的內容。
一般的情況開發人員是很反感這種文字性的描述的,往往是因為開發人員習慣于對待硬梆梆的PC機,而不愿意去面對客戶,收集這種需求。其實文字性的東西,既是一個對系統的概述,又是我們發現開發要素的土壤。試想如果你的軟件要發布了,你卻無法組織起語言讓用戶恰當的理解軟件的功能與使用對象,是多么的讓人煩惱。
精練用戶的需求(其實是我的教學需求哈哈)。很顯然添加記錄與修改記錄同屬于對記錄進行編輯操作,就叫做編輯記錄用例(NoteEdit)好了。瀏覽記錄也是一個很明顯的用例,就叫做瀏覽用例(Notepad)。刪除是對記錄進行的一種管理,叫做管理記錄用例(NoteManager)。到此,我們已發現并精練了三個主要用例,還不錯,系統正一步步變的清晰。在這里提醒大家,這個階段是站在客戶的觀點(這里是用戶的觀點)想問題的,你的javascript:;" onClick="javascript:tagshow(event, '%B9%A4%D7%F7');" target="_self">工作是發現并系統化客戶的想法,不必站在開發者的角度思考任何細節。
So,let ‘go。
編輯記錄用例(NoteEdit)
事件流1:
1) 顯示用戶Note的內容
2) 用戶編輯內容
3) 用戶放棄修改,note內容不變,正常退出
事件流2:
1) 顯示用戶Note的內容
2) 用戶編輯內容
3) 用戶save,退回主菜單
事件流3:
1) 顯示用戶Note的內容
2) 用戶編輯內容
3) 用戶save As,提示讓用戶輸入新的文件名
4) Save,退回主菜單
瀏覽用例(Notepad)
事件流1:
1) 顯示用戶的Note的標題、創建時間、內容
2) 用戶選擇退出,返回主菜單
事件流2:
1) 顯示用戶的Note的標題、創建時間、內容
2) 用戶顯示編輯,轉向編輯用例
管理記錄用例(NoteManager)
事件流1:
1) 顯示用戶的Note列表
2) 用戶打開選擇的Note,轉向瀏覽用例
事件流2:
1) 顯示用戶的Note列表
2) 用戶編輯選擇的Note,轉向編輯用例
事件流3:
1) 顯示用戶的Note列表
2) 用戶新建一個Note,轉向編輯用例
事件流4:
1) 顯示用戶的Note列表
2) 用戶刪除選擇的Note
3) 出現確認提示
4) 用戶確認,刪除Note
5) 更新顯示,回到Note列表
三個用例的事件流一經被分析出來了,很顯然應該在第一次迭代全部完成。
尋找類(oo分析)首先是實體類(Entity),只需要從事件流中提取名字就可以縮小范圍。
Note,顯然是個對象。
內容(content)、時間(datetime)、標題(title),恩,應該是Note的元素。
Note的方法包括對域成員的操作set/get。因為要保存,所以需要序列化反序列化方法。
一般實體類都是由一個對應的生命周期類(lifecycle)用于他的產生、存儲、消亡等等操作,一般把這樣的操作獨立出來大大有利用實體類的重用。不過此階段還用不著分析他,一會兒畫順序圖時,自然就會發現它。
習慣上為了高效的畫順序圖,邊界類和控制類的方法都需一一列出。不過我們省了,大家只知道每個用例都對應著一個邊界類就好了。
設計實踐我舉三個用例中管理記錄用例(NoteManager)的一部分和瀏覽用例(Notepad)的一部分來介紹詳細的設計過程。在這里,我們試圖從一個客戶的角度轉化到一個開發者角度。要面對很多的挑戰,可能包括一部分細節。應該學習從分離的角度思考整個系統。MVC的精華就在這里。
記錄用例(NoteManager)事件流1:
NoteManagerUI并不知道Note列表的具體組織形式,它通過預先商定好的接口getNoteTitleList向控制類NoteManagerWorkflow所要數據,控制類返回一個String[]數組。
同樣,NoteManagerWorkflow需要向生命周期類NoteLocator所要數據,不過NoteManagerWorkflow知道數據的細節。為了能夠識別數據,除了返回記錄的Title這一信息外,還要同時返回一個唯一識別的ID作為整個系統內識別Note的方法。所以NoteManagerWorkflow就有了兩個域一個是TitleList、一個是IdList。
這里有幾個細節:
1) NoteManagerUI、NoteManagerWorkflow如何通信,這不成問題,我們有理由相信他們是緊密相關的。
2) NoteManagerWorkflow如何找到NoteLocator,一般情況下,NoteLoator都是單件Singlton。
3) 當NoteManagerUI的showNote(index),調用的時候,他會調用NoteManager的showNoteDispose(index),而showNoteDispose會根據內部的實現,將這一Index轉化為id用于識別Note
瀏覽用例(Notepad)事件流1:
NotepadUI向控制類所要標題,控制類有域note、和noteid,但是控制類通過getNode,這里很明顯的可以使用惰性初始化技術,向生命周期類所要Note。取得記錄對象的引用后,你可以方便的像實體類請求數據了。
經驗分享當然了,打好骨架后你就可以開始時coding了,畫圖的好處是強迫你在設計階段做好各個部件之間的接口設計。這可以有效地減少你返工的幾率,但是往往我們在設計階段過多的思考了細節,比如NoteLocator是如何和Rms交流的等等。這都是很不好的習慣,但是不太容易改正。因為無論是學校里,還是陪訓等等都是訓練,反復的訓練我們對coding的敏感。我們太依賴于從代碼的角度思考問題了,這阻礙了我們從大局思考問題,發現更通用的模式。
如果要開始coding了,也不要一開始就全面鋪開,一般實體類具有很強的獨立性??梢元毩㈤_發,而開發其它類的時候可以從邊界類開始,如果想一邊開發一邊測試一下,大可把控制類、生命周期類的方法暫時用fade data(偽支撐數據),這都是大大降低復雜性的好辦法。
惰性初始化是我使用的最為頻繁的技術,我覺得它可以大大降低代碼混亂的程度。
整個系統的UML有人說看不懂UML,但如果給我這么大的代碼,我肯定看不懂。UML嗎,有可能看懂:)
屏幕快照開始后的畫面:(管理用例)
新建后的畫面:(編輯用例)
按下save,輸入title名字
更新顯示:
選擇你喜歡的瀏覽,比如標題是easy的記錄:(瀏覽用例)
屏幕導航:代碼種種 |
原文轉自:http://www.anti-gravitydesign.com