王亞萍 張紅霞
(河南農(nóng)業(yè)職業(yè)學(xué)院,河南 中牟451450)
MVC 設(shè)計(jì)模式: 是一種架構(gòu)型設(shè)計(jì)模式, 它本身不引入新的功能,只是指導(dǎo)我們把Web 應(yīng)用結(jié)構(gòu)做的更加合理,實(shí)現(xiàn)邏輯與頁面相分離。
功能型設(shè)計(jì)模式:如單例(保證類的實(shí)例唯一)、工廠(選擇實(shí)現(xiàn))、值對象(封裝數(shù)據(jù))、DAO(屏蔽變化)等等。
架構(gòu)型設(shè)計(jì)模式:結(jié)構(gòu)性的設(shè)計(jì)模式,只是讓程序更具有結(jié)構(gòu)化。
Java 中MVC 來源思路及進(jìn)化過程:Servlet =Java + HTML 字符串,問題產(chǎn)生了,由于拼字符串太麻煩→解決方案:HTML 獨(dú)立出來從而得到→JSP≈HTML+Java 腳本(問題又產(chǎn)生了,能解決Servlet 問題,但是帶來了頁面和邏輯混雜)→解決方案:MVC(指導(dǎo)我們讓W(xué)eb 應(yīng)用程序結(jié)構(gòu)更加合理)。
①M(fèi)odel:VO+邏輯層:可以理解為后臺(tái)部分。
功能:封裝應(yīng)用狀態(tài)、響應(yīng)狀態(tài)查詢、暴露應(yīng)用功能。
②View:視圖層的三大功能,典型的如修改頁面,觸發(fā)事件而不處理,純jsp 自己處理,提交給Controller 處理。
功能:產(chǎn)生HTML 響應(yīng)、請求模型更新、提供HTML 表單用于用戶請求。
對比: 純jsp 頁面中事件處理與頁面展示混雜在一起,MVC 中事件處理由Controller 承擔(dān),從而達(dá)到邏輯與頁面相分離的效果。
③Controller:事件處理過程。
功能:驗(yàn)證HTML 請求的數(shù)據(jù)、將用戶數(shù)據(jù)與模型更新相映射、選擇用于響應(yīng)的視圖。
View 用戶請求到控制器,控制器狀態(tài)改變通知Model,Model 主動(dòng)通知View 說Model 自身已改變,View 主動(dòng)去Model 里面去狀態(tài)查詢。
下面先看個(gè)標(biāo)準(zhǔn)的MVC 單機(jī)版的示例:
Eclipse 本身就是基于MVC 做的,例如當(dāng)我們打開Eclipse 時(shí)修改編輯區(qū)的代碼時(shí),左邊的Navigator 視圖和右邊的Outline 視圖等都會(huì)自動(dòng)更新而隨著編輯區(qū)的代碼改變而改變。實(shí)際上這些窗口觀察的都是同一個(gè)Model,即觀察XX.java 代碼文件的內(nèi)容,這種多個(gè)View 觀察者觀察同一個(gè)Model 文件的在設(shè)計(jì)模式中也有個(gè)設(shè)計(jì)模式與其對應(yīng)即觀察者設(shè)計(jì)模式。
觀察者設(shè)計(jì)模式Java 代碼示例的如下:
(1)根據(jù)前面的內(nèi)容抽象,首先必須有個(gè)被觀察的對象,即目標(biāo)對象叫做MySubject.java:
觀察者觀察的為目標(biāo)對象的內(nèi)容Content, 當(dāng)內(nèi)容改變了即有人setContent 需要通知所有的觀察者。
(2)其次應(yīng)該有些觀察者,叫做MyObserver:
MyObserver 實(shí)現(xiàn)一個(gè)update 回調(diào)方法, 即MyObserver 去觀察MySubject,觀察到后采取的處理即這個(gè)update 方法。 這里的觀察者又有兩種模型,一種是被觀察者主動(dòng)推過來的消息即推模型,另一種就是觀察者主動(dòng)去拉內(nèi)容即拉模型。 其實(shí)觀察者模式也叫出版訂閱模式,推模型即訂報(bào)的意思,即報(bào)刊發(fā)行商將報(bào)紙送上門服務(wù),拉模型即某個(gè)時(shí)刻大家主動(dòng)去報(bào)刊亭買報(bào)的這種模型。
(3)客戶端測試代碼,Client.java:
首先創(chuàng)建一個(gè)目標(biāo)對象,然后創(chuàng)建三個(gè)觀察者,然后注冊觀察者相當(dāng)于訂報(bào)紙,然后報(bào)社出報(bào)紙即內(nèi)容改變時(shí)觸發(fā)觀察者,當(dāng)被觀察的對象改變時(shí)需要通知觀察者也即對應(yīng)MySubject.java。
代碼輸出結(jié)果:
wwu 推過來的====null
wwu 主動(dòng)去拉====觀察者模式
lsi 推過來的====null
lsi 主動(dòng)去拉====觀察者模式
當(dāng)將Client.java 文件中的MyObserver ob2=new MyObserver("lsi");、和subject.addObserver(ob2);注釋掉時(shí),輸出結(jié)果中間2 行隨之也不顯示出來,即所謂的lis 退訂。
③改進(jìn)的版的MVC
在Java WEB 開發(fā)中會(huì)產(chǎn)生一個(gè)問題: 就是為什么Model 會(huì)主動(dòng)通知View 自身已經(jīng)改變了呢?因?yàn)镴ava 中標(biāo)準(zhǔn)的MVC 起源于Swing,這種標(biāo)準(zhǔn)的MVC 只能適應(yīng)于單機(jī)版,在WEB 開發(fā)中無法實(shí)現(xiàn),因?yàn)閃EB 是基于請求應(yīng)答模式的環(huán)境,Model 不會(huì)在沒有請求的情況下直接通知View 自身的改變。
在實(shí)際開發(fā)WEB 應(yīng)用的時(shí)候, 由于無法按照標(biāo)準(zhǔn)的MVC 去實(shí)現(xiàn),通常我們會(huì)把邏輯部分轉(zhuǎn)移到邏輯層去實(shí)現(xiàn),所以Model 就退變成只用來封裝數(shù)據(jù),也就是我們常寫的VO;此時(shí)View 一般不直接和邏輯層交互,所有跟邏輯層的交互都由控制器來實(shí)現(xiàn),View 只和控制器交互。
改進(jìn)版后的MVC 一般實(shí)現(xiàn)方式:View 用JSP 來實(shí)現(xiàn),Controller由Servlet 來實(shí)現(xiàn),Model 由JavaBean 來實(shí)現(xiàn)。
[1][美]Robert Lafore.計(jì)曉云,趙研,等譯.Java 數(shù)據(jù)結(jié)構(gòu)與算法[M].北京:中國電力出版社,2003.