過(guò)去的時(shí)間中牵啦,支撐開(kāi)發(fā)的話(huà)題一直都是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)的是封裝在類(lèi)中作瞄,這也是迎合OO思想的表現(xiàn)茶宵。當(dāng)然,也會(huì)有放進(jìn)特殊數(shù)據(jù)結(jié)構(gòu)中的做法宗挥,但對(duì)于OO來(lái)說(shuō)乌庶,這些數(shù)據(jù)結(jié)構(gòu)也是類(lèi)(當(dāng)然,如果這么說(shuō)是在用抽象的思想解決實(shí)際問(wèn)題契耿,沒(méi)有意義瞒大。)?。
實(shí)際的來(lái)看搪桂,數(shù)據(jù)最多情況下放在了自己定義的類(lèi)中糠赦,包括VO,PO,DAO等。
PO:持久對(duì)象锅棕,是持久層使用的對(duì)象拙泽。目前使用PO對(duì)數(shù)據(jù)庫(kù)查詢(xún)條件進(jìn)行過(guò)封裝,這個(gè)對(duì)象使用的意義在于裸燎,將數(shù)據(jù)庫(kù)查詢(xún)語(yǔ)句中需要的數(shù)據(jù)裝在一個(gè)對(duì)象中顾瞻,這樣調(diào)用java中執(zhí)行sql語(yǔ)句的方法時(shí),只需要向參數(shù)中傳入該對(duì)象即可德绿,避免出現(xiàn)參數(shù)過(guò)多的情況荷荤。
DAO:數(shù)據(jù)訪(fǎng)問(wèn)對(duì)象。主要是用對(duì)象的思維來(lái)對(duì)數(shù)據(jù)查詢(xún)進(jìn)行封裝移稳,其中包含的主要是對(duì)數(shù)據(jù)的不同操作的抽象蕴纳,并在其中提供實(shí)現(xiàn)數(shù)據(jù)操作的方法。通過(guò)DAO个粱,可以實(shí)現(xiàn)數(shù)據(jù)的真實(shí)查詢(xún)古毛。
VO:數(shù)據(jù)對(duì)象。主要是用面向?qū)ο蟮姆绞椒庋b一個(gè)數(shù)據(jù)塊都许,其中包括對(duì)數(shù)據(jù)進(jìn)行描述的POJO(簡(jiǎn)單java類(lèi)型)對(duì)象稻薇,在類(lèi)中被看作是字段,這些字段如果沒(méi)有g(shù)et或set方法胶征,在web應(yīng)用中幾乎沒(méi)有什么用處塞椎。當(dāng)他們獲得了自己的getter或setter的時(shí)候,這些字段被稱(chēng)作“屬性”睛低。屬性是相對(duì)于javabean而言的案狠。javabean中可以包含多種不同的POJO類(lèi)型服傍。當(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框架依賴(lài)的數(shù)據(jù)庫(kù)連接技術(shù)耍贾,配置中包含有數(shù)據(jù)庫(kù)位置阅爽,JDBC核心類(lèi)引用以及數(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)封裝能扒,方法是佣渴,利用反射從指定的類(lèi)中去找到屬性,對(duì)屬性(或字段)進(jìn)行賦值初斑。此處辛润,需要用來(lái)封裝數(shù)據(jù)的類(lèi)的字段或?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)代碼被瀏覽器解釋?zhuān)┑奶攸c(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)格以及客戶(hù)配置約束钻注,進(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ù)類(lèi),不同的行就是這個(gè)數(shù)據(jù)類(lèi)的不同實(shí)例對(duì)象热监。?對(duì)于前端代碼而言捺弦,使用struts提供的標(biāo)簽,直接通過(guò)數(shù)據(jù)的別名可以對(duì)這些數(shù)據(jù)進(jìn)行操作孝扛。對(duì)于后端代碼而言列吼,只要在struts2中提供了這些數(shù)據(jù)對(duì)應(yīng)的getter或setter,就可以按照使用類(lèi)的方式對(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)完成了功能霎挟。