過(guò)去的時(shí)間中,支撐開(kāi)發(fā)的話題一直都是CRUD,雖然形式多樣,但無(wú)非就這四個(gè)操作。由于沒(méi)有涉及到業(yè)務(wù),基本上也就都是些簡(jiǎn)單技術(shù)的使用。雖然技術(shù)都是簡(jiǎn)單的,沒(méi)有設(shè)計(jì)模式,沒(méi)有復(fù)雜邏輯,沒(méi)有業(yè)務(wù)邏輯,但由于是框架封裝,數(shù)據(jù)在其中的流動(dòng)和各個(gè)地方的存在形式依然需要好好記錄。
先畫(huà)張圖:
這里的整理只是已知的部分,還有很多細(xì)節(jié)沒(méi)有體現(xiàn)。更靠譜的說(shuō)法是,這是張思維導(dǎo)圖。
首先看簡(jiǎn)單的部分,中間的service層。
由于service層的代碼結(jié)構(gòu),或者是數(shù)據(jù)結(jié)構(gòu)都是根據(jù)需求不停改變,所以,(在已知的情況中)沒(méi)有一個(gè)好的框架對(duì)這一層進(jìn)行好的封裝。所以數(shù)據(jù)流動(dòng)也更容易看見(jiàn),畢竟都是在自己寫(xiě)的代碼中,數(shù)據(jù)存在方式也是java中經(jīng)典的數(shù)據(jù)存儲(chǔ)方式,最常見(jiàn)的是封裝在類中,這也是迎合OO思想的表現(xiàn)。當(dāng)然,也會(huì)有放進(jìn)特殊數(shù)據(jù)結(jié)構(gòu)中的做法,但對(duì)于OO來(lái)說(shuō),這些數(shù)據(jù)結(jié)構(gòu)也是類(當(dāng)然,如果這么說(shuō)是在用抽象的思想解決實(shí)際問(wèn)題,沒(méi)有意義。)?。
實(shí)際的來(lái)看,數(shù)據(jù)最多情況下放在了自己定義的類中,包括VO,PO,DAO等。
PO:持久對(duì)象,是持久層使用的對(duì)象。目前使用PO對(duì)數(shù)據(jù)庫(kù)查詢條件進(jìn)行過(guò)封裝,這個(gè)對(duì)象使用的意義在于,將數(shù)據(jù)庫(kù)查詢語(yǔ)句中需要的數(shù)據(jù)裝在一個(gè)對(duì)象中,這樣調(diào)用java中執(zhí)行sql語(yǔ)句的方法時(shí),只需要向參數(shù)中傳入該對(duì)象即可,避免出現(xiàn)參數(shù)過(guò)多的情況。
DAO:數(shù)據(jù)訪問(wèn)對(duì)象。主要是用對(duì)象的思維來(lái)對(duì)數(shù)據(jù)查詢進(jìn)行封裝,其中包含的主要是對(duì)數(shù)據(jù)的不同操作的抽象,并在其中提供實(shí)現(xiàn)數(shù)據(jù)操作的方法。通過(guò)DAO,可以實(shí)現(xiàn)數(shù)據(jù)的真實(shí)查詢。
VO:數(shù)據(jù)對(duì)象。主要是用面向?qū)ο蟮姆绞椒庋b一個(gè)數(shù)據(jù)塊,其中包括對(duì)數(shù)據(jù)進(jìn)行描述的POJO(簡(jiǎn)單java類型)對(duì)象,在類中被看作是字段,這些字段如果沒(méi)有g(shù)et或set方法,在web應(yīng)用中幾乎沒(méi)有什么用處。當(dāng)他們獲得了自己的getter或setter的時(shí)候,這些字段被稱作“屬性”。屬性是相對(duì)于javabean而言的。javabean中可以包含多種不同的POJO類型。當(dāng)形成javabean的時(shí)候,在整個(gè)項(xiàng)目中的數(shù)據(jù)流動(dòng)就有了基礎(chǔ)。
雖然在service層提到了javabean,但是數(shù)據(jù)的數(shù)據(jù)封裝并不是在這里形成的。真實(shí)的數(shù)據(jù)存在于“持久層”中,數(shù)據(jù)的封裝也是在這里形成的。
現(xiàn)在來(lái)看持久層。
從命名就可以看出,這一層是將數(shù)據(jù)永久保存起來(lái)的地方。數(shù)據(jù)永久保存的方式就在于把數(shù)據(jù)寫(xiě)進(jìn)掉電可存儲(chǔ)的硬件中,對(duì)于個(gè)人計(jì)算機(jī)來(lái)講,就是硬盤(pán)。但是數(shù)據(jù)不是隨便誰(shuí)都可以去管理的,因此需要一個(gè)數(shù)據(jù)管理的工具。同時(shí),數(shù)據(jù)如果是零散的也沒(méi)有意義,需要一個(gè)有序的數(shù)據(jù)存儲(chǔ)方式(數(shù)據(jù)結(jié)構(gòu)),這些數(shù)據(jù)存儲(chǔ)方式管理和數(shù)據(jù)管理(數(shù)據(jù)導(dǎo)入,建立數(shù)學(xué)模型等等),通過(guò)“數(shù)據(jù)庫(kù)”來(lái)實(shí)現(xiàn)。
既然是有序的來(lái)實(shí)現(xiàn)數(shù)據(jù)存儲(chǔ),就必定有一套規(guī)則來(lái)進(jìn)行描述。數(shù)據(jù)庫(kù)中有自己的描述方式,而java中有另一種描述方式。想要讓java中的數(shù)據(jù)能夠很好的放進(jìn)數(shù)據(jù)庫(kù)中,需要用到ORM(對(duì)象關(guān)系映射)。
想要實(shí)現(xiàn)這套規(guī)則,并為實(shí)現(xiàn)這套規(guī)則建立恰當(dāng)?shù)膶?shí)現(xiàn)環(huán)境(數(shù)據(jù)庫(kù)連接池,數(shù)據(jù)庫(kù)連接,對(duì)象拆裝,數(shù)據(jù)封裝等等),并不容易,但同樣,每次要做的操作又很相似。為此,出現(xiàn)了一系列框架,這里用的是Mybatis框架。
框架使用的配置文件包括,?數(shù)據(jù)庫(kù)連接池配置,數(shù)據(jù)庫(kù)JDBC配置,mapper配置。
數(shù)據(jù)連接池配置:Mybatis用來(lái)建議的配置連接池的文件,一旦配置文件成功引用,框架將自動(dòng)創(chuàng)建連接池,與數(shù)據(jù)庫(kù)保持長(zhǎng)期連接并管理數(shù)據(jù)連接資源。
數(shù)據(jù)庫(kù)JDBC配置:JDBC配置。JSBC是Mybatis框架依賴的數(shù)據(jù)庫(kù)連接技術(shù),配置中包含有數(shù)據(jù)庫(kù)位置,JDBC核心類引用以及數(shù)據(jù)庫(kù)連接的賬號(hào)和密碼。
mapper配置:這個(gè)配置文件中,寫(xiě)的全是sql語(yǔ)句,同時(shí),該配置文件在需要的時(shí)候,還要用來(lái)指定數(shù)據(jù)封裝的格式。對(duì)于沒(méi)有配置封裝格式的數(shù)據(jù),Mybatis將實(shí)現(xiàn)自動(dòng)封裝,方法是,利用反射從指定的類中去找到屬性,對(duì)屬性(或字段)進(jìn)行賦值。此處,需要用來(lái)封裝數(shù)據(jù)的類的字段或?qū)傩悦蛿?shù)據(jù)庫(kù)中列的名字相對(duì)應(yīng)。?(框架約定,遵循該規(guī)則的情況下,不需要對(duì)數(shù)據(jù)進(jìn)行更多配置)
至此,數(shù)據(jù)的獲得得到了保障。從統(tǒng)一的數(shù)據(jù)源中獲得數(shù)據(jù)也按照我們需要的格式進(jìn)行了數(shù)據(jù)模型的表達(dá)。
數(shù)據(jù)從持久層中獲得并成功封裝以后,通過(guò)方法返回值、對(duì)象使用或者靜態(tài)變量使用等方式,服務(wù)層也可以獲得這些?數(shù)據(jù),并進(jìn)行進(jìn)一步處理。處理結(jié)束以后,數(shù)據(jù)繼續(xù)通過(guò)與持久層相同的方式,向展示層的后臺(tái)代碼中流動(dòng)。
展示層:
展示層只做一件事情,就是展示數(shù)據(jù)。這也是結(jié)果看起來(lái)相對(duì)不那么枯燥,但其實(shí)也比較麻煩的一個(gè)地方。
首先,要知道的是,除去展示的前段代碼,包括JSP,HTML等等,剩下的部分都是寫(xiě)在Java中的代碼,也就是跑在應(yīng)用服務(wù)器中的代碼,?算作是后臺(tái)代碼。由于后臺(tái)代碼和前臺(tái)代碼有物理上分開(kāi)(后臺(tái)代碼在服務(wù)器中,前臺(tái)代碼被瀏覽器解釋)的特點(diǎn),數(shù)據(jù)流動(dòng)需要經(jīng)過(guò)一些轉(zhuǎn)換。在不涉及底層的情況下,可以認(rèn)為,struts2的值棧充當(dāng)了數(shù)據(jù)中轉(zhuǎn)的機(jī)制。(底層實(shí)現(xiàn)方式,可以猜想為,通過(guò)http協(xié)議,將數(shù)據(jù)進(jìn)行某種格式放在http請(qǐng)求或響應(yīng)中,在網(wǎng)絡(luò)環(huán)境中傳輸。對(duì)于前端來(lái)說(shuō),數(shù)據(jù)被瀏覽器通過(guò)http協(xié)議解析以后,拼湊成完整的文件,并對(duì)不同的文件按照自己的格式解析出來(lái),再通過(guò)某種編程規(guī)范,將數(shù)據(jù)重新拼湊,形成一長(zhǎng)段有價(jià)值的數(shù)據(jù)流。這段數(shù)據(jù)流被瀏覽器按照國(guó)際規(guī)范以及自己瀏覽器的獨(dú)有風(fēng)格以及客戶配置約束,進(jìn)行一系列復(fù)雜的編譯,最終變成了另一種有意義的數(shù)據(jù)形式,放進(jìn)計(jì)算機(jī)相應(yīng)的硬件中,最終通過(guò)計(jì)算機(jī)硬件進(jìn)行展示。對(duì)于后臺(tái),同樣需要對(duì)http協(xié)議指定格式的數(shù)據(jù)進(jìn)行解析,這一步發(fā)生在JVM中,解析以后的數(shù)據(jù)按照我們事先在代碼中約定的方式進(jìn)行封裝以后就可以被程序猿使用了。)
對(duì)于值棧,其實(shí)就是提供了數(shù)據(jù)在不同物理位置,但是相同表現(xiàn)形式的可能。按照編程中透明化的要求,對(duì)于后臺(tái)開(kāi)發(fā)程序猿而言,值棧就是可以被后臺(tái)和前臺(tái)都認(rèn)識(shí)的一種特殊的數(shù)據(jù)存儲(chǔ)。?值棧中有兩部分,分別是ValueStackContents和StackContext。
ValueStackContent是值棧中數(shù)值存儲(chǔ)的位置。其中包含的所有棧行都可以看作是值棧中封裝的一個(gè)個(gè)數(shù)據(jù)類,不同的行就是這個(gè)數(shù)據(jù)類的不同實(shí)例對(duì)象。?對(duì)于前端代碼而言,使用struts提供的標(biāo)簽,直接通過(guò)數(shù)據(jù)的別名可以對(duì)這些數(shù)據(jù)進(jìn)行操作。對(duì)于后端代碼而言,只要在struts2中提供了這些數(shù)據(jù)對(duì)應(yīng)的getter或setter,就可以按照使用類的方式對(duì)數(shù)據(jù)進(jìn)行使用。
StackContext運(yùn)行上下文。值棧中用來(lái)保存上下文數(shù)據(jù)的位置,其中通過(guò)Map的形式對(duì)數(shù)據(jù)進(jìn)行表述,每條數(shù)據(jù)都對(duì)應(yīng)自己的一個(gè)鍵,前端代碼通過(guò)OGNL表達(dá)式對(duì)數(shù)據(jù)進(jìn)行引用,后端代碼通過(guò)對(duì)map進(jìn)行操作對(duì)數(shù)據(jù)進(jìn)行讀寫(xiě)
(ActionContext.getContext().get(String key); ? OR ? ActionContext.getContext().put(String key,Object value);)。
至此,數(shù)據(jù)在struts2中(展示層)中的存儲(chǔ)和使用已經(jīng)結(jié)束。綜合從服務(wù)層的數(shù)據(jù)流動(dòng),數(shù)據(jù)完成了從數(shù)據(jù)庫(kù)中的提取,服務(wù)層的處理和轉(zhuǎn)發(fā),到達(dá)了展示層,按照五花八門(mén)的方式展示出來(lái)。
對(duì)于基本的web應(yīng)用而言,已經(jīng)完成了功能。