欧美色欧美亚洲高清在线观看,国产特黄特色a级在线视频,国产一区视频一区欧美,亚洲成a 人在线观看中文

  1. <ul id="fwlom"></ul>

    <object id="fwlom"></object>

    <span id="fwlom"></span><dfn id="fwlom"></dfn>

      <object id="fwlom"></object>

      java-論壇總結(jié)前人思考貼,認(rèn)真入門的同學(xué)要有耐心看完

      時(shí)間:2019-05-14 04:18:38下載本文作者:會(huì)員上傳
      簡(jiǎn)介:寫寫幫文庫(kù)小編為你整理了多篇相關(guān)的《java-論壇總結(jié)前人思考貼,認(rèn)真入門的同學(xué)要有耐心看完》,但愿對(duì)你工作學(xué)習(xí)有幫助,當(dāng)然你在寫寫幫文庫(kù)還可以找到更多《java-論壇總結(jié)前人思考貼,認(rèn)真入門的同學(xué)要有耐心看完》。

      第一篇:java-論壇總結(jié)前人思考貼,認(rèn)真入門的同學(xué)要有耐心看完

      http://blog.csdn.net/ant_yan[/b]

      [img=http://hi.csdn.net/attachment/201003/17/1442_12687963784zxy.jpg][/img]

      想來(lái)學(xué)習(xí)Java也有兩個(gè)年頭了,永遠(yuǎn)不敢說多么精通,但也想談?wù)勛约旱母惺埽瑢懡o軟件學(xué)院的同仁們,幫助大家在技術(shù)的道路上少一點(diǎn)彎路。說得偉大一點(diǎn)是希望大家為軟件學(xué)院爭(zhēng)氣,其實(shí)最主要的還是大家自身的進(jìn)步提升——

      Java學(xué)習(xí)雜談一

      1. 關(guān)于動(dòng)態(tài)加載機(jī)制——

      學(xué)習(xí)Java比C++更容易理解OOP的思想,畢竟C++還混合了不少面向過程的成分。很多人都能背出來(lái)Java語(yǔ)言的特點(diǎn),所謂的動(dòng)態(tài)加載機(jī)制等等。當(dāng)然概念往往是先記住而后消化的,可有多少人真正去體會(huì)過動(dòng)態(tài)加載的機(jī)制,試圖去尋找過其中的細(xì)節(jié)呢? 提供大家一個(gè)方法:

      在命令行窗口運(yùn)行Java程序的時(shí)候,加上這個(gè)很有用的參數(shù):

      java –verbose *.class

      這樣會(huì)清晰的打印出被加載的類文件,大部分是jdk自身運(yùn)行需要的,最后幾行會(huì)明顯的看到自己用到的那幾個(gè)類文件被加載進(jìn)來(lái)的順序。即使你聲明了一個(gè)類對(duì)象,不實(shí)例化也不會(huì)加載,說明只有真正用到那個(gè)類的實(shí)例即對(duì)象的時(shí)候,才會(huì)執(zhí)行加載。這樣是不是大家稍微能明白一點(diǎn)動(dòng)態(tài)加載了呢?^_^

      2. 關(guān)于尋找class文件原理——

      建議大家在入門的時(shí)候在命令行窗口編譯和運(yùn)行,不要借助JCreator或者Eclipse等IDE去幫助做那些事情。嘗試自己這樣做:

      javac

      -classpath yourpath *.java java

      -classpath yourpath *.class

      也許很多人都能看懂,設(shè)置classpath的目的就是告訴編譯器去哪里尋找你的class文件.不過至少筆者今日才弄懂JVM去查詢類的原理,編譯器加載類要依靠classloader,而classloader有3個(gè)級(jí)別,從高到低分別是BootClassLoader(名字可能不準(zhǔn)確), ExtClassLoader, AppClassLoader.這3個(gè)加載器分別對(duì)應(yīng)著編譯器去尋找類文件的優(yōu)先級(jí)別和不同的路徑:BootClassLoader對(duì)應(yīng)jre/classes路徑,是編譯器最優(yōu)先尋找class的地方

      ExtClassLoader對(duì)應(yīng)jre/lib/ext路徑,是編譯器次優(yōu)先尋找class的地方

      AppClassLoader對(duì)應(yīng)當(dāng)前路徑,所以也是編譯器默認(rèn)找class的地方

      其實(shí)大家可以自己寫個(gè)程序簡(jiǎn)單的測(cè)試,對(duì)任何class,例如A,調(diào)用new A().getClass().getClassLoader().toString()打印出來(lái)就可以看到,把class文件放在不同的路徑下再次執(zhí)行,就會(huì)看到區(qū)別。特別注意的是如果打印出來(lái)是null就表示到了最高級(jí)BootClassLoader, 因?yàn)樗荂++編寫的,不存在Java對(duì)應(yīng)的類加載器的名字。

      尋找的順序是一種向上迂回的思想,即如果本級(jí)別找不到,就只能去本級(jí)別之上的找,不會(huì)向下尋找。不過似乎從Jdk1.4到Jdk1.6這一特點(diǎn)又有改變,沒有找到詳細(xì)資料。所以就不舉例子了。告訴大家設(shè)計(jì)這種體系的是Sun公司曾經(jīng)的技術(shù)核心宮力先生,一個(gè)純種華人哦!^_^

      這樣希望大家不至于迷惑為什么總報(bào)錯(cuò)找不到類文件,不管是自己寫的還是導(dǎo)入的第三方的jar文件(J2ee中經(jīng)常需要導(dǎo)入的)。

      3. 關(guān)于jdk和jre——

      大家肯定在安裝JDK的時(shí)候會(huì)有選擇是否安裝單獨(dú)的jre,一般都會(huì)一起安裝,我也建議大家這樣做。因?yàn)檫@樣更能幫助大家弄清楚它們的區(qū)別:

      Jre 是java runtime environment, 是java程序的運(yùn)行環(huán)境。既然是運(yùn)行,當(dāng)然要包含jvm,也就是大家熟悉的虛擬機(jī)啦,還有所有java類庫(kù)的class文件,都在lib目錄下打包成了jar。大家可以自己驗(yàn)證。至于在windows上的虛擬機(jī)是哪個(gè)文件呢?

      學(xué)過MFC的都知道什么是dll文件吧,那么大家看看jre/bin/client里面是不是有一個(gè)jvm.dll呢?那就是虛擬機(jī)。

      Jdk 是java development kit,是java的開發(fā)工具包,里面包含了各種類庫(kù)和工具。當(dāng)然也包括了另外一個(gè)Jre.那么為什么要包括另外一個(gè)Jre呢?而且jdk/jre/bin同時(shí)有client和server兩個(gè)文件夾下都包含一個(gè)jvm.dll。

      說明是有兩個(gè)虛擬機(jī)的。這一點(diǎn)不知道大家是否注意到了呢?

      相信大家都知道jdk的bin下有各種java程序需要用到的命令,與jre的bin目錄最明顯的區(qū)別就是jdk下才有javac,這一點(diǎn)很好理解,因?yàn)閖re只是一個(gè)運(yùn)行環(huán)境而已。與開發(fā)無(wú)關(guān),正因?yàn)槿绱?,具備開發(fā)功能的jdk自己的jre下才會(huì)同時(shí)有client性質(zhì)的jvm和server性質(zhì)的jvm,而僅僅作為運(yùn)行環(huán)境的jre下只需要client性質(zhì)的jvm.dll就夠了。

      記得在環(huán)境變量path中設(shè)置jdk/bin路徑麼?這應(yīng)該是大家學(xué)習(xí)Java的第一步吧,老師會(huì)告訴大家不設(shè)置的話javac和java是用不了的。確實(shí)jdk/bin目錄下包含了所有的命令??墒怯袥]有人想過我們用的java命令并不是jdk/bin目錄下的而是jre/bin目錄下的呢?不信可以做一個(gè)實(shí)驗(yàn),大家可以把jdk/bin目錄下的java.exe剪切到別的地方再運(yùn)行java程序,發(fā)現(xiàn)了什么?一切OK!

      那么有人會(huì)問了?我明明沒有設(shè)置jre/bin目錄到環(huán)境變量中?。?/p>

      試想一下如果java為了提供給大多數(shù)人使用,他們是不需要jdk做開發(fā)的,只需要jre能讓java程序跑起來(lái)就可以了,那么每個(gè)客戶還需要手動(dòng)去設(shè)置環(huán)境變量多麻煩???所以安裝jre的時(shí)候安裝程序自動(dòng)幫你把jre的java.exe添加到了系統(tǒng)變量中,驗(yàn)證的方法很簡(jiǎn)單,大家看到了系統(tǒng)環(huán)境變量的path最前面有―%SystemRoot%system32;%SystemRoot%;‖這樣的配置,那么再去Windows/system32下面去看看吧,發(fā)現(xiàn)了什么?有一個(gè)java.exe。

      如果強(qiáng)行能夠把jdk/bin挪到system32變量前面,當(dāng)然也可以迫使使用jdk/jre里面的java,不過除非有必要,我不建議大家這么做。使用單獨(dú)的jre跑java程序也算是客戶環(huán)境下的一種測(cè)試。

      這下大家應(yīng)該更清楚jdk和jre內(nèi)部的一些聯(lián)系和區(qū)別了吧?

      PS: 其實(shí)還有滿多感想可以總結(jié)的,一次寫多了怕大家扔磚頭砸死我,怪我太羅唆。大家應(yīng)該更加踏實(shí)更加務(wù)實(shí)的去做一些研究并互相分享心得,大方向和太前沿的技術(shù)討論是必要的但最好不要太多,畢竟自己基礎(chǔ)都還沒打好,什么都講最新版本其實(shí)是進(jìn)步的一大障礙!

      Java 學(xué)習(xí)雜談

      (二)鑒于上回寫的一點(diǎn)感想大家不嫌棄,都鼓勵(lì)小弟繼續(xù)寫下去,好不容易等到國(guó)慶黃金周,實(shí)習(xí)總算有一個(gè)休息的階段,于是這就開始寫第二篇了。希望這次寫的仍然對(duì)志同道合的朋友們有所幫助。上回講了Java動(dòng)態(tài)加載機(jī)制、classLoader原理和關(guān)于jdk和jre三個(gè)問題。這次延續(xù)著講一些具體的類庫(kù)——

      1. 關(guān)于集合框架類

      相信學(xué)過Java的各位對(duì)這個(gè)名詞并不陌生,對(duì) java.util.*這個(gè)package肯定也不陌生。不知道大家查詢API的時(shí)候怎么去審視或者分析其中的一個(gè)package,每個(gè)包最重要的兩個(gè)部分就是interfaces和classes,接口代表了它能做什么,實(shí)現(xiàn)類則代表了它如何去做。關(guān)注實(shí)現(xiàn)類之前,我們應(yīng)該先理解清楚它的來(lái)源接口,不管在j2se還是j2ee中,都應(yīng)該是這樣。那么我們先看這三個(gè)接口:List、Set、Map。

      也許有些人不太熟悉這三個(gè)名字,但相信大部分人都熟悉ArrayList,LinkedList,TreeSet,HashSet,HashMap,Hashtable等實(shí)現(xiàn)類的名字。它們的區(qū)別也是滿容易理解的,List放可以重復(fù)的對(duì)象集合,Set放不可重復(fù)的對(duì)象組合,而Map則放 這樣的名值對(duì),Key不可重復(fù),Value可以。這里有幾個(gè)容易混淆的問題:

      到底Vector和ArrayList,Hashtable和HashMap有什么區(qū)別?

      很多面試官喜歡問這個(gè)問題,其實(shí)更專業(yè)一點(diǎn)應(yīng)該這樣問:新集合框架和舊集合框架有哪些區(qū)別?新集合框架大家可以在這些包中找since jdk1.2的,之前的如vector和Hashtable都是舊的集合框架包括的類。那么區(qū)別是?

      a.新集合框架的命名更加科學(xué)合理。例如List下的ArrayList和LinkedList

      b.新集合框架下全部都是非線程安全的。建議去jdk里面包含的源代碼里面自己去親自看看vector和ArrayList的區(qū)別吧。當(dāng)然如果是jdk5.0之后的會(huì)比較難看一點(diǎn),因?yàn)橛旨尤肓朔盒偷恼Z(yǔ)法,類似c++的template語(yǔ)法。

      那么大家是否想過為什么要從舊集合框架默認(rèn)全部加鎖防止多線程訪問更新到新集合框架全部取消鎖,默認(rèn)方式支持多線程?(當(dāng)然需要的時(shí)候可以使用collections的靜態(tài)方法加鎖達(dá)到線程安全)

      筆者的觀點(diǎn)是任何技術(shù)的發(fā)展都未必是遵循它們的初衷的,很多重大改變是受到客觀環(huán)境的影響的。大家知道Java的初衷是為什么而開發(fā)的麼?是為嵌入式程序開發(fā)的。記得上一篇講到classLoader機(jī)制麼?那正是為了節(jié)約嵌入式開發(fā)環(huán)境下內(nèi)存而設(shè)計(jì)的。而走到今天,Java成了人們心中為互聯(lián)網(wǎng)誕生的語(yǔ)言。互聯(lián)網(wǎng)意味著什么?多線程是必然的趨勢(shì)??陀^環(huán)境在變,Java技術(shù)也隨著飛速發(fā)展,導(dǎo)致越來(lái)越脫離它的初衷。據(jù)說Sun公司其實(shí)主打的是J2se,結(jié)果又是由于客觀環(huán)境影響,J2se幾乎遺忘,留在大家談?wù)摻裹c(diǎn)的一直是j2ee。

      技術(shù)的細(xì)節(jié)這里就不多說了,只有用了才能真正理解。解釋這些正是為了幫助大家理解正在學(xué)的和將要學(xué)的任何技術(shù)。之后講j2ee的時(shí)候還會(huì)再討論。

      多扯句題外話:幾十年前的IT巨人是IBM,Mainframe市場(chǎng)無(wú)人可比。微軟如何打敗IBM?正是由于硬件飛速發(fā)展,對(duì)個(gè)人PC的需求這個(gè)客觀環(huán)境,讓微軟通過OS稱為了第二個(gè)巨人。下一個(gè)打敗微軟的呢?Google。如何做到的?如果微軟并不和IBM爭(zhēng)大型機(jī),Google借著互聯(lián)網(wǎng)飛速發(fā)展這個(gè)客觀環(huán)境作為決定性因素,避開跟微軟爭(zhēng)OS,而是走搜索引擎這條路,稱為第3個(gè)巨人。那么第4個(gè)巨人是誰(shuí)呢?很多專家預(yù)言將在亞洲或者中國(guó)出現(xiàn),Whatever,客觀環(huán)境變化趨勢(shì)才是決定大方向的關(guān)鍵。當(dāng)然筆者也希望會(huì)出現(xiàn)在中國(guó),^_^~~

      2. 關(guān)于Java設(shè)計(jì)模式

      身邊的很多在看GOF的23種設(shè)計(jì)模式,似乎學(xué)習(xí)它無(wú)論在學(xué)校還是在職場(chǎng),都成了一種流行風(fēng)氣。我不想列舉解釋這23種Design Pattern,我寫這些的初衷一直都是談自己的經(jīng)歷和看法,希望能幫助大家理解。

      首先我覺得設(shè)計(jì)模式只是對(duì)一類問題的一種通用解決辦法,只要是面向?qū)ο蟮木幊填A(yù)言都可以用得上這23種。理解它們最好的方法就是親自去寫每一種,哪怕是一個(gè)簡(jiǎn)單的應(yīng)用就足夠了。如果代碼實(shí)現(xiàn)也記不住的話,記憶它們對(duì)應(yīng)的UML圖會(huì)是一個(gè)比較好的辦法,當(dāng)然前提是必須了解UML。

      同時(shí)最好能利用Java自身的類庫(kù)幫助記憶,例如比較常用的觀察者模式,在java.util.*有現(xiàn)成的Observer接口和Observable這個(gè)實(shí)現(xiàn)類,看看源代碼相信就足夠理解觀察者模式了。再比如裝飾器模式,大家只要寫幾個(gè)關(guān)于java.io.*的程序就可以完全理解什么是裝飾器模式了。有很多人覺得剛?cè)腴T的時(shí)候不該接觸設(shè)計(jì)模式,比如圖靈設(shè)計(jì)叢書系列很出名的那本《Java設(shè)計(jì)模式》,作者: Steven John Metsker,大部分例子老實(shí)說令現(xiàn)在的我也很迷惑。但我仍然不同意入門跟學(xué)習(xí)設(shè)計(jì)模式有任何沖突,只是我們需要知道每種模式的概念的和典型的應(yīng)用,這樣我們?cè)诘谝淮尉帉?FileOutputStream、BufferedReader、PrintWriter的時(shí)候就能感覺到原來(lái)設(shè)計(jì)模式離我們?nèi)绱酥也⒉皇嵌嗝瓷衩氐臇|西。

      另外,在學(xué)習(xí)某些模式的同時(shí),反而更能幫助我們理解java類庫(kù)的某些特點(diǎn)。例如當(dāng)你編寫原型(Prototype)模式的時(shí)候,你必須了解的是 java.lang.Cloneable這個(gè)接口和所有類的基類Object的clone()這個(gè)方法。即深copy和淺copy的區(qū)別:

      Object.clone()默認(rèn)實(shí)現(xiàn)的是淺copy,也就是復(fù)制一份對(duì)象拷貝,但如果對(duì)象包含其他對(duì)象的引用,不會(huì)復(fù)制引用,所以原對(duì)象和拷貝共用那個(gè)引用的對(duì)象。

      深copy當(dāng)然就是包括對(duì)象的引用都一起復(fù)制啦。這樣原對(duì)象和拷貝對(duì)象,都分別擁有一份引用對(duì)象。如果要實(shí)現(xiàn)深copy就必須首先實(shí)現(xiàn) java.lang.Cloneable接口,然后重寫clone()方法。因?yàn)樵贠bject中的clone()方法是protected簽名的,而 Cloneable接口的作用就是把protected放大到public,這樣clone()才能被重寫。

      那么又有個(gè)問題了?如果引用的對(duì)象又引用了其他對(duì)象呢?這樣一直判斷并復(fù)制下去,是不是顯得很麻煩?曾經(jīng)有位前輩告訴我的方法是重寫clone方法的時(shí)候直接把原對(duì)象序列化到磁盤上再反序列化回來(lái),這樣不用判斷就可以得到一個(gè)深copy的結(jié)果。如果大家不了解序列化的作法建議看一看 ObjectOutputStream和ObjectInputStream

      歸根結(jié)底,模式只是思想上的東西,把它當(dāng)成前人總結(jié)的經(jīng)驗(yàn)其實(shí)一點(diǎn)都不為過。鼓勵(lì)大家動(dòng)手自己去寫,例如代理模式,可以簡(jiǎn)單的寫一個(gè)Child類,Adult類。Child要買任何東西由Adult來(lái)代理實(shí)現(xiàn)。簡(jiǎn)單來(lái)說就是Adult里的buy()內(nèi)部實(shí)際調(diào)用的是Child的buy(),可是暴露在main函數(shù)的卻是Adult.buy()。這樣一個(gè)簡(jiǎn)單的程序就足夠理解代理模式的基本含義了。

      Java 雜談

      (三)這已經(jīng)筆者寫的第三篇Java雜記了,慶幸前兩篇一直得到論壇朋友們的支持鼓勵(lì),還望大家繼續(xù)指正不足之處。筆者也一直渴望通過這樣方式清醒的自審,來(lái)尋找自己技術(shù)上的不足之處,希望和共同愛好Java的同仁們一起提高。

      前兩次分別講述了關(guān)于jvm、jdk、jre、collection、classLoader和一些Design Pattern的自我理解。這次仍然不準(zhǔn)備開始過渡到j(luò)2ee中,因?yàn)橛X得還有一些瑣碎的j2se的問題沒有總結(jié)完畢。

      1. 關(guān)于Object類理解

      大家都知道Object是所有Java類的基類,意味著所有的Java類都會(huì)繼承了Object的11個(gè)方法。建議大家去看看Object的 11個(gè)成員函數(shù)的源代碼,就會(huì)知道默認(rèn)的實(shí)現(xiàn)方式。比如equals方法,默認(rèn)實(shí)現(xiàn)就是用“==”來(lái)比較,即直接比較內(nèi)存地址,返回true 或者 false。而toString()方法,返回的串組成方式是——

      “getClass().getName()+ ”@“ + Integer.toHexString(hashCode())”

      其實(shí)不用我過多的解釋,大家都能看懂這個(gè)串的組成。接下來(lái)再看看hashCode(): public native int hashCode();

      由于是native方法,跟OS的處理方式相關(guān),源代碼里僅僅有一個(gè)聲明罷了。我們有興趣的話完全可以去深究它的hashCode到底是由OS怎么樣產(chǎn)生的呢?但筆者建議最重要的還是先記住使用它的幾條原則吧!首先如果equals()方法相同的對(duì)象具有相通的hashCode,但equals()對(duì)象不相通的時(shí)候并不保證hashCode()方法返回不同的整數(shù)。而且下一次運(yùn)行同一個(gè)程序,同一個(gè)對(duì)象未必還是當(dāng)初的那個(gè)hashCode()哦。

      其余的方法呢?nofigy()、notifyAll()、clone()、wait()都是native方法的,說明依賴于操作系統(tǒng)的實(shí)現(xiàn)。最后一個(gè)有趣的方法是finalize(),類似C++的析構(gòu)函數(shù),簽名是protected,證明只有繼承擴(kuò)展了才能使用,方法體是空的,默示什么也不做。它的作用據(jù)筆者的了解僅僅是通知JVM此對(duì)象不再使用,隨時(shí)可以被銷毀,而實(shí)際的銷毀權(quán)還是在于虛擬機(jī)手上。那么它真的什么也不做麼?未必,實(shí)際上如果是線程對(duì)象它會(huì)導(dǎo)致在一定范圍內(nèi)該線程的優(yōu)先級(jí)別提高,導(dǎo)致更快的被銷毀來(lái)節(jié)約內(nèi)存提高性能。其實(shí)從常理來(lái)說,我們也可以大概這樣猜測(cè)出jvm做法的目的。

      2. 關(guān)于重載hashCode()與Collection框架的關(guān)系

      筆者曾經(jīng)聽一位搞Java培訓(xùn)多年的前輩說在他看來(lái)hashCode方法沒有任何意義,僅僅是為了配合證明具有同樣的hashCode會(huì)導(dǎo)致equals 方法相等而存在的。連有的前輩都犯這樣的錯(cuò)誤,其實(shí)說明它還是滿容易被忽略的。那么hashCode()方法到底做什么用?

      學(xué)過數(shù)據(jù)結(jié)構(gòu)的課程大家都會(huì)知道有一種結(jié)構(gòu)叫hash table,目的是通過給每個(gè)對(duì)象分配一個(gè)唯一的索引來(lái)提高查詢的效率。那么Java也不會(huì)肆意扭曲改變這個(gè)概念,所以hashCode唯一的作用就是為支持?jǐn)?shù)據(jù)結(jié)構(gòu)中的哈希表結(jié)構(gòu)而存在的,換句話說,也就是只有用到集合框架的 Hashtable、HashMap、HashSet的時(shí)候,才需要重載hashCode()方法,這樣才能使得我們能人為的去控制在哈希結(jié)構(gòu)中索引是否相等。筆者舉一個(gè)例子:

      曾經(jīng)為了寫一個(gè)求解類程序,需要隨機(jī)列出1,2,3,4組成的不同排列組合,所以筆者寫了一個(gè)數(shù)組類用int[]來(lái)存組合結(jié)果,然后把隨機(jī)產(chǎn)生的組合加入一個(gè)HashSet中,就是想利用HashSet不包括重復(fù)元素的特點(diǎn)??墒荋ashSet怎么判斷是不是重復(fù)的元素呢?當(dāng)然是通過 hashCode()返回的結(jié)果是否相等來(lái)判斷啦,可做一下這個(gè)實(shí)驗(yàn):

      int[] A = {1,2,3,4};int[] B = {1,2,3,4};

      System.out.println(A.hashCode());System.out.println(B.hashCode());

      這明明是同一種組合,卻是不同的hashCode,加入Set的時(shí)候會(huì)被當(dāng)成不同的對(duì)象。這個(gè)時(shí)候我們就需要自己來(lái)重寫hashCode()方法了,如何寫呢?其實(shí)也是基于原始的hashCode(),畢竟那是操作系統(tǒng)的實(shí)現(xiàn),找到相通對(duì)象唯一的標(biāo)識(shí),實(shí)現(xiàn)方式很多,筆者的實(shí)現(xiàn)方式是:

      首先重寫了toString()方法:

      return A[0]―+‖ A[1]―+‖ A[2]―+‖ A[3];//顯示上比較直觀

      然后利用toString()來(lái)計(jì)算hashCode():

      return this.toString().hashCode();

      這樣上述A和B返回的就都是‖1234‖,在測(cè)試toString().hashCode(),由于String在內(nèi)存中的副本是一樣的,‖1234‖.hashCode()返回的一定是相同的結(jié)果。

      說到這,相信大家能理解得比我更好,今后千萬(wàn)不要再誤解hashCode()方法的作用。

      3. 關(guān)于Class類的成員函數(shù)與Java反射機(jī)制

      很早剛接觸Java就聽很多老師說過Java的動(dòng)態(tài)運(yùn)行時(shí)機(jī)制、反射機(jī)制等。確實(shí)它們都是Java的顯著特點(diǎn),運(yùn)行時(shí)加載筆者在第一篇介紹過了,現(xiàn)在想講講反射機(jī)制。在Java中,主要是通過java.lang包中的Class類和Method類來(lái)實(shí)現(xiàn)內(nèi)存反射機(jī)制的。

      熟悉C++的人一定知道下面這樣在C++中是做不到的: 運(yùn)行時(shí)以字符串參數(shù)傳遞一個(gè)類名,就可以得到這個(gè)類的所有信息,包括它所有的方法,和方法的詳細(xì)信息。還可以實(shí)例化一個(gè)對(duì)象,并通過查到的方法名來(lái)調(diào)用該對(duì)象的任何方法。這是因?yàn)镴ava的類在內(nèi)存中除了C++中也有的靜態(tài)動(dòng)態(tài)數(shù)據(jù)區(qū)之外,還包括一份對(duì)類自身的描述,也正是通過這描述中的信息,才能幫助我們才運(yùn)行時(shí)讀取里面的內(nèi)容,得到需要加載目標(biāo)類的所有信息,從而實(shí)現(xiàn)反射機(jī)制。大家有沒有想過當(dāng)我們需要得到一個(gè)JavaBean的實(shí)例的時(shí)候,怎么知道它有哪些屬性呢?再明顯簡(jiǎn)單不過的例子就是自己寫一個(gè)JavaBean的解析器:

      a.通過Class.forName(―Bean的類名‖)得到Class對(duì)象,例如叫ABeanClass b.通過ABeanClass的getMethods()方法,得到Method[]對(duì)象

      c.按照規(guī)范所有g(shù)et方法名后的單詞就代表著該Bean的一個(gè)屬性

      d.當(dāng)已經(jīng)知道一個(gè)方法名,可以調(diào)用newInstance()得到一個(gè)實(shí)例,然后通過invoke()方法將方法的名字和方法需要用的參數(shù)傳遞進(jìn)去,就可以動(dòng)態(tài)調(diào)用此方法。

      當(dāng)然還有更復(fù)雜的應(yīng)用,這里就不贅述,大家可以參考Class類和Method類的方法。

      4. 坦言Synchronize的本質(zhì)

      Synchronize大家都知道是同步、加鎖的意思,其實(shí)它的本質(zhì)遠(yuǎn)沒有大家想得那么復(fù)雜。聲明Synchronize的方法被調(diào)用的時(shí)候,鎖其實(shí)是加載對(duì)象上,當(dāng)然如果是靜態(tài)類則是加在類上的鎖,調(diào)用結(jié)束鎖被解除。它的實(shí)現(xiàn)原理很簡(jiǎn)單,僅僅是不讓第二把鎖再次被加在同一個(gè)對(duì)象或類上,僅此而已。一個(gè)簡(jiǎn)單的例子足以說明問題:

      class A{

      synchronized void f(){} void g(){} }

      當(dāng)A的一個(gè)對(duì)象a被第一個(gè)線程調(diào)用其f()方法的時(shí)候,第二個(gè)線程不能調(diào)用a的synchronized方法例如f(),因?yàn)槟鞘窃谠噲D在對(duì)象上加第二把鎖。但調(diào)用g()卻是可以的,因?yàn)椴]有在同一對(duì)象上加兩把鎖的行為產(chǎn)生。

      這樣大家能理解了麼?明白它的原理能更好的幫助大家設(shè)計(jì)同步機(jī)制,不要濫用加鎖。

      PS:下篇筆者計(jì)劃開始對(duì)J2ee接觸到的各個(gè)方面來(lái)進(jìn)行總結(jié),談?wù)勛约旱慕?jīng)驗(yàn)和想法。希望大家還能一如既往的支持筆者寫下去,指正不足之處。

      Java雜談

      (四)不知不覺已經(jīng)寫到第四篇了,論壇里面不斷的有朋友鼓勵(lì)我寫下去。堅(jiān)持自己的作風(fēng),把一切迷惑不容易理清楚的知識(shí)講出來(lái),講到大家都能聽懂,那么自己就真的懂了。最近在公司實(shí)習(xí)的時(shí)候Trainer跟我講了很多經(jīng)典事跡,對(duì)還未畢業(yè)的我來(lái)說是筆不小的財(cái)富,我自己的信念是:人在逆境中成長(zhǎng)的速度要遠(yuǎn)遠(yuǎn)快過順境中,這樣來(lái)看一切都能欣然接受了。

      好了,閑話不說了,第三篇講的是反射機(jī)制集合框架之類的,這次打算講講自己對(duì)反序列化和多線程的理解。希望能對(duì)大家學(xué)習(xí)Java起到幫助——

      1.關(guān)于序列化和反序列化

      應(yīng)該大家都大概知道Java中序列化和反序列化的意思,序列化就是把一個(gè)Java對(duì)象轉(zhuǎn)換成二進(jìn)制進(jìn)行磁盤上傳輸或者網(wǎng)絡(luò)流的傳輸,反序列化的意思就是把這個(gè)接受到的二進(jìn)制流重新組裝成原來(lái)的對(duì)象逆過程。它們?cè)贘ava中分別是通過ObjectInputStream和 ObjectInputStream這兩個(gè)類來(lái)實(shí)現(xiàn)的(以下分別用ois和oos來(lái)簡(jiǎn)稱)。

      oos的writeObject()方法用來(lái)執(zhí)行序列化的過程,ois的readObject()用來(lái)執(zhí)行反序列化的過程,在傳輸二進(jìn)制流之前,需要講這兩個(gè)高層流對(duì)象連接到同一個(gè)Channel上,這個(gè)Channel可以是磁盤文件,也可以是socket底層流。所以無(wú)論用哪種方式,底層流對(duì)象都是以構(gòu)造函數(shù)參數(shù)的形式傳遞進(jìn)oos和ois這兩個(gè)高層流,連接完畢了才可以進(jìn)行二進(jìn)制數(shù)據(jù)傳輸?shù)?。例子?/p>

      可以是文件流通道

      file = new File(―C:/data.dat‖);

      oos = new ObjectOutputStream(new FileOutputStream(file));ois = new ObjectInputStream(new FileInputStream(file));

      或者網(wǎng)絡(luò)流通道

      oos = new ObjectOutputStream(socket.getOutputStream());ois = new ObjectInputStream(socket.getInputStream());

      不知道大家是否注意到oos總是在ois之前定義,這里不希望大家誤解這個(gè)順序是固定的么?回答是否定的,那么有順序要求么?回答是肯定的。原則是什么呢?

      原則是互相對(duì)接的輸入/輸出流之間必須是output流先初始化然后再input流初始化,否則就會(huì)拋異常。大家肯定會(huì)問為什么?只要稍微看一看這兩個(gè)類的源代碼文件就大概知道了,output流的任務(wù)很簡(jiǎn)單,只要把對(duì)象轉(zhuǎn)換成二進(jìn)制往通道中寫就可以了,但input流需要做很多準(zhǔn)備工作來(lái)接受并最終重組這個(gè)Object,所以O(shè)bjectInputStream的構(gòu)造函數(shù)中就需要用到output初始化發(fā)送過來(lái)的header信息,這個(gè)方法叫做 readStreamHeader(),它將會(huì)去讀兩個(gè)Short值用于決定用多大的緩存來(lái)存放通道發(fā)送過來(lái)的二進(jìn)制流,這個(gè)緩存的size因jre的版本不同是不一樣的。所以output如果不先初始化,input的構(gòu)造函數(shù)首先就無(wú)法正確運(yùn)行。

      對(duì)于上面兩個(gè)例子,第一個(gè)順序是嚴(yán)格的,第二個(gè)因?yàn)閛os和ois連接的已經(jīng)不是對(duì)方了,而是socket另外一端的流,需要嚴(yán)格按照另外一方對(duì)接的output流先于對(duì)接的input流打開才能順利運(yùn)行。

      這個(gè)writeObject和readObject本身就是線程安全的,傳輸過程中是不允許被并發(fā)訪問的。所以對(duì)象能一個(gè)一個(gè)接連不斷的傳過來(lái),有很多人在運(yùn)行的時(shí)候會(huì)碰到EOFException, 然后百思不得其解,去各種論壇問解決方案。其實(shí)筆者這里想說,這個(gè)異常不是必須聲明的,也就是說它雖然是異常,但其實(shí)是正常運(yùn)行結(jié)束的標(biāo)志。EOF表示讀到了文件尾,發(fā)送結(jié)束自然連接也就斷開了。如果這影響到了你程序的正確性的話,請(qǐng)各位靜下心來(lái)看看自己程序的業(yè)務(wù)邏輯,而不要把注意力狹隘的聚集在發(fā)送和接受的方法上。因?yàn)楣P者也被這樣的bug困擾了1整天,被很多論壇的帖子誤解了很多次最后得出的教訓(xùn)。如果在while循環(huán)中去readObject,本質(zhì)上是沒有問題的,有對(duì)象數(shù)據(jù)來(lái)就會(huì)讀,沒有就自動(dòng)阻塞。那么拋出EOFException一定是因?yàn)檫B接斷了還在繼續(xù)read,什么原因?qū)е逻B接斷了呢?一定是業(yè)務(wù)邏輯哪里存在錯(cuò)誤,比如NullPoint、ClassCaseException、ArrayOutofBound,即使程序較大也沒關(guān)系,最多只要單步調(diào)適一次就能很快發(fā)現(xiàn)bug并且解決它。

      難怪一位程序大師說過:解決問題90%靠經(jīng)驗(yàn),5%靠技術(shù),剩下5%靠運(yùn)氣!真是金玉良言,筆者大概查閱過不下30篇討論在while循環(huán)中使用 readObject拋出EOFExceptionde 的帖子,大家都盲目的去關(guān)注解釋這個(gè)名詞、反序列化的行為或反對(duì)這樣寫而沒有一個(gè)人認(rèn)為EOF是正確的行為,它其實(shí)很老實(shí)的在做它的事情。為什么大家都忽略了真正出錯(cuò)誤的地方呢??jī)蓚€(gè)字,經(jīng)驗(yàn)!

      2.關(guān)于Java的多線程編程

      關(guān)于Java的線程,初學(xué)或者接觸不深的大概也能知道一些基本概念,同時(shí)又會(huì)很迷惑線程到底是怎么回事?如果有人認(rèn)為自己已經(jīng)懂了不妨來(lái)回答下面的問題:

      a.A對(duì)象實(shí)現(xiàn)Runnable接口,A.start()運(yùn)行后所謂的線程對(duì)象是誰(shuí)?是A么?

      b.線程的wait()、notify()方法到底是做什么時(shí)候用的,什么時(shí)候用?

      c.為什么線程的suspend方法會(huì)被標(biāo)注過時(shí),不推薦再使用,線程還能掛起么?

      d.為了同步我們會(huì)對(duì)線程方法聲明Synchronized來(lái)加鎖在對(duì)象上,那么如果父類的f()方法加了Synchronized,子類重寫f()方法必須也加Synchronized么?如果子類的f()方法重寫時(shí)聲明Synchronized并調(diào)用super.f(),那么子類對(duì)象上到底有幾把鎖呢?會(huì)因?yàn)楦?jìng)爭(zhēng)產(chǎn)生死鎖么?

      呵呵,各位能回答上來(lái)幾道呢?如果這些都能答上來(lái),說明對(duì)線程的概念還是滿清晰的,雖說還遠(yuǎn)遠(yuǎn)不能算精通。筆者這里一一做回答,礙于篇幅的原因,筆者盡量說得簡(jiǎn)介一點(diǎn),如果大家有疑惑的歡迎一起討論。

      首先第一點(diǎn),線程跟對(duì)象完全是兩回事,雖然我們也常說線程對(duì)象。但當(dāng)你用run()和start()來(lái)啟動(dòng)一個(gè)線程之后,線程其實(shí)跟這個(gè)繼承了 Thread或?qū)崿F(xiàn)了Runnable的對(duì)象已經(jīng)沒有關(guān)系了,對(duì)象只能算內(nèi)存中可用資源而對(duì)象的方法只能算內(nèi)存正文區(qū)可以執(zhí)行的代碼段而已。既然是資源和代碼段,另外一個(gè)線程當(dāng)然也可以去訪問,main函數(shù)執(zhí)行就至少會(huì)啟動(dòng)兩個(gè)線程,一個(gè)我們稱之為主線程,還一個(gè)是垃圾收集器的線程,主線程結(jié)束就意味著程序結(jié)束,可垃圾收集器線程很可能正在工作。

      第二點(diǎn),wait()和sleep()類似,都是讓線程處于阻塞狀態(tài)暫停一段時(shí)間,不同之處在于wait會(huì)釋放當(dāng)前線程占有的所有的鎖,而 sleep不會(huì)。我們知道獲得鎖的唯一方法是進(jìn)入了Synchronized保護(hù)代碼段,所以大家會(huì)發(fā)現(xiàn)只有Synchronized方法中才會(huì)出現(xiàn) wait,直接寫會(huì)給警告沒有獲得當(dāng)前對(duì)象的鎖。所以notify跟wait配合使用,notify會(huì)重新把鎖還給阻塞的線程重而使其繼續(xù)執(zhí)行,當(dāng)有多個(gè)對(duì)象wait了,notify不能確定喚醒哪一個(gè),必經(jīng)鎖只有一把,所以一般用notifyAll()來(lái)讓它們自己根據(jù)優(yōu)先級(jí)等競(jìng)爭(zhēng)那唯一的一把鎖,競(jìng)爭(zhēng)到的線程執(zhí)行,其他線程只要繼續(xù)wait。

      從前Java允許在一個(gè)線程之外把線程掛起,即調(diào)用suspend方法,這樣的操作是極不安全的。根據(jù)面向?qū)ο蟮乃枷朊總€(gè)對(duì)象必須對(duì)自己的行為負(fù)責(zé),而對(duì)自己的權(quán)力進(jìn)行封裝。如果任何外步對(duì)象都能使線程被掛起而阻塞的話,程序往往會(huì)出現(xiàn)混亂導(dǎo)致崩潰,所以這樣的方法自然是被斃掉了啦。

      最后一個(gè)問題比較有意思,首先回答的是子類重寫f()方法可以加Synchronized也可以不加,如果加了而且還內(nèi)部調(diào)用了super.f()的話理論上是應(yīng)該對(duì)同一對(duì)象加兩把鎖的,因?yàn)槊看握{(diào)用Synchronized方法都要加一把,調(diào)用子類的f首先就加了一把,進(jìn)入方法內(nèi)部調(diào)用父類的 f又要加一把,加兩把不是互斥的么?那么調(diào)父類f加鎖不就必須永遠(yuǎn)等待已經(jīng)加的鎖釋放而造成死鎖么?實(shí)際上是不會(huì)的,這個(gè)機(jī)制叫重進(jìn)入,當(dāng)父類的f方法試圖在本對(duì)象上再加一把鎖的時(shí)候,因?yàn)楫?dāng)前線程擁有這個(gè)對(duì)象的鎖,也可以理解為開啟它的鑰匙,所以同一個(gè)線程在同一對(duì)象上還沒釋放之前加第二次鎖是不會(huì)出問題的,這個(gè)鎖其實(shí)根本就沒有加,它有了鑰匙,不管加幾把還是可以進(jìn)入鎖保護(hù)的代碼段,暢通無(wú)阻,所以叫重進(jìn)入,我們可以簡(jiǎn)單認(rèn)為第二把鎖沒有加上去。

      總而言之,Synchronized的本質(zhì)是不讓其他線程在同一對(duì)象上再加一把鎖。

      [size=5]Java雜談(五)[/size]

      本來(lái)預(yù)計(jì)J2se只講了第四篇就收尾了,可是版主厚愛把帖子置頂長(zhǎng)期讓大家瀏覽讓小弟倍感責(zé)任重大,務(wù)必追求最到更好,所以關(guān)于J2se一些沒有提到的部分,決定再寫幾篇把常用的部分經(jīng)驗(yàn)全部寫出來(lái)供大家討論切磋。這一篇準(zhǔn)備講一講Xml解析包和Java Swing,然后下一篇再講java.security包關(guān)于Java沙箱安全機(jī)制和RMI機(jī)制,再進(jìn)入J2ee的部分,暫時(shí)就做這樣的計(jì)劃了。如果由于實(shí)習(xí)繁忙更新稍微慢了一些,希望各位見諒!

      1. Java關(guān)于XML的解析

      相信大家對(duì)XML都不陌生,含義是可擴(kuò)展標(biāo)記語(yǔ)言。本身它也就是一個(gè)數(shù)據(jù)的載體以樹狀表現(xiàn)形式出現(xiàn)。后來(lái)慢慢的數(shù)據(jù)變成了信息,區(qū)別是信息可以包括可變的狀態(tài)從而針對(duì)程序硬編碼的做法變革為針對(duì)統(tǒng)一接口硬編碼而可變狀態(tài)作為信息進(jìn)入了XML中存儲(chǔ)。這樣改變狀態(tài)實(shí)現(xiàn)擴(kuò)展的唯一工作是在XML中添加一段文本信息就可以了,代碼不需要改動(dòng)也不需要重新編譯。這個(gè)靈活性是XML誕生時(shí)候誰(shuí)也沒想到的。

      當(dāng)然,如果接口要能提取XML中配置的信息就需要程序能解析規(guī)范的XML文件,Java中當(dāng)然要提高包對(duì)這個(gè)行為進(jìn)行有利支持。筆者打算講到的兩個(gè)包是org.w3c.dom和javax.xml.parsers和。(大家可以瀏覽一下這些包中間的接口和類定義)

      Javax.xml.parsers包很簡(jiǎn)單,沒有接口,兩個(gè)工廠配兩個(gè)解析器。顯然解析XML是有兩種方式的:DOM解析和SAX解析。本質(zhì)上并沒有誰(shuí)好誰(shuí)不好,只是實(shí)現(xiàn)的思想不一樣罷了。給一個(gè)XML文件的例子:

      A Cat

      所謂DOM解析的思路是把整個(gè)樹狀圖存入內(nèi)存中,需要那個(gè)節(jié)點(diǎn)只需要在樹上搜索就可以讀到節(jié)點(diǎn)的屬性,內(nèi)容等,這樣的好處是所有節(jié)點(diǎn)皆在內(nèi)存可以反復(fù)搜索重復(fù)使用,缺點(diǎn)是需要消耗相應(yīng)的內(nèi)存空間。

      自然SAX解析的思路就是為了克服DOM的缺點(diǎn),以事件觸發(fā)為基本思路,順序的搜索下來(lái),碰到了Element之前觸發(fā)什么事件,碰到之后做什么動(dòng)作。由于需要自己來(lái)寫觸發(fā)事件的處理方案,所以需要借助另外一個(gè)自定義的Handler,處于org.xml.sax.helpers包中。它的優(yōu)點(diǎn)當(dāng)然是不用整個(gè)包都讀入內(nèi)存,缺點(diǎn)也是只能順序搜索,走完一遍就得重來(lái)。

      大家很容易就能猜到,接觸到的J2ee框架用的是哪一種,顯然是DOM。因?yàn)轭愃芐truts,Hibernate框架配置文件畢竟是很小的一部分配置信息,而且需要頻繁搜索來(lái)讀取,當(dāng)然會(huì)采用DOM方式(其實(shí)SAX內(nèi)部也是用DOM采用的結(jié)構(gòu)來(lái)存儲(chǔ)節(jié)點(diǎn)信息的)?,F(xiàn)在無(wú)論用什么框架,還真難發(fā)現(xiàn)使用SAX來(lái)解析XML的技術(shù)了,如果哪位仁兄知道,請(qǐng)讓筆者也學(xué)習(xí)學(xué)習(xí)。

      既然解析方式有了,那么就需要有解析的存儲(chǔ)位置。不知道大家是否發(fā)現(xiàn)org.w3c.dom這個(gè)包是沒有實(shí)現(xiàn)類全部都是接口的。這里筆者想說一下Java如何對(duì)XML解析是Jdk應(yīng)該考慮的事,是它的責(zé)任。而w3c組織是維護(hù)定義XML標(biāo)準(zhǔn)的組織,所以一個(gè)XML結(jié)構(gòu)是怎么樣的由w3c說了算,它不關(guān)心Java如何去實(shí)現(xiàn),于是乎規(guī)定了所有XML存儲(chǔ)的結(jié)構(gòu)應(yīng)該遵循的規(guī)則,這就是org.w3c.dom里全部的接口目的所在。在筆者看來(lái),簡(jiǎn)單理解接口的概念就是實(shí)現(xiàn)者必須遵守的原則。

      整個(gè)XML對(duì)應(yīng)的結(jié)構(gòu)叫Document、子元素對(duì)應(yīng)的叫做Element、還有節(jié)點(diǎn)相關(guān)的Node、NodeList、Text、Entity、CharacterData、CDATASection等接口,它們都可以在XML的語(yǔ)法中間找到相對(duì)應(yīng)的含義。由于這里不是講解XML基本語(yǔ)法,就不多介紹了。如果大家感興趣,筆者也可以專門寫一篇關(guān)于XML的語(yǔ)法規(guī)則帖與大家分享一下。

      2. Java Swing

      Swing是一個(gè)讓人又愛又恨的東西,可愛之處在于上手很容易,較AWT比起來(lái)Swing提供的界面功能更加強(qiáng)大,可恨之處在于編復(fù)雜的界面工作量實(shí)在是巨大。筆者寫過超過3000行的Swing界面,感覺用戶體驗(yàn)還不是那么優(yōu)秀。最近又寫過超過6000行的,由于功能模塊多了,整體效果還只是一般般。體會(huì)最深的就一個(gè)字:累!所以大家現(xiàn)在都陸續(xù)不怎么用Swing在真正開發(fā)的項(xiàng)目上了,太多界面技術(shù)可以取代它了。筆者去寫也是迫于無(wú)奈組里面大家都沒寫過,我不入地域誰(shuí)入?

      盡管Swing慢慢的在被人忽略,特別是隨著B/S慢慢的在淹沒C/S,筆者倒是很愿意站出來(lái)為Swing正身。每一項(xiàng)技術(shù)的掌握絕不是為了流行時(shí)尚跟風(fēng)。真正喜歡Java的朋友們還是應(yīng)該好好體會(huì)一下Swing,相信在校的很多學(xué)生也很多在學(xué)習(xí)它。很可能從Jdk 1.1、1.2走過來(lái)的很多大學(xué)老師可能是最不熟悉它的。

      Swing提供了一組輕組件統(tǒng)稱為JComponent,它們與AWT組件的最大區(qū)別是JComponent全部都是Container,而Container的特點(diǎn)是里面可以裝載別的組件。在Swing組件中無(wú)論是JButton、JLabel、JPanel、JList等都可以再裝入任何其他組件。好處是程序員可以對(duì)Swing組件實(shí)現(xiàn)―再開發(fā)‖,針對(duì)特定需求構(gòu)建自己的按鈕、標(biāo)簽、畫板、列表之類的特定組件。

      有輕自然就有重,那么輕組件和重組件區(qū)別是?重組件表現(xiàn)出來(lái)的形態(tài)因操作系統(tǒng)不同而異,輕組件是Swing自己提供GUI,在跨平臺(tái)的時(shí)候最大程度的保持一致。

      那么在編程的時(shí)候要注意一些什么呢?筆者談?wù)勛约旱膸c(diǎn)經(jīng)驗(yàn):

      a.明確一個(gè)概念,只有Frame組件才可以單獨(dú)顯示的,也許有人會(huì)說JOptionPane里面的靜態(tài)方法就實(shí)現(xiàn)了單獨(dú)窗口出現(xiàn),但追尋源代碼會(huì)發(fā)現(xiàn)其實(shí)現(xiàn)實(shí)出來(lái)的Dialog也需要依托一個(gè)Frame窗體,如果沒有指定就會(huì)默認(rèn)產(chǎn)生一個(gè)然后裝載這個(gè)Dialog顯示出來(lái)。

      b.JFrame是由這么幾部分組成:

      最底下一層JRootPane,上面是glassPane(一個(gè)JPanel)和layeredPane(一個(gè)JLayeredPane),而layeredPane又由contentPane(一個(gè)JPanel)和menuBar構(gòu)成。我們的組件都是加在contentPane上,而背景圖片只能加在layeredPane上面。至于glassPane是一個(gè)透明的覆蓋了contentPane的一層,在特定效果中將被利用到來(lái)記錄鼠標(biāo)坐標(biāo)或掩飾組件。

      c.為了增強(qiáng)用戶體驗(yàn),我們會(huì)在一些按鈕上添加快捷鍵,但Swing里面通常只能識(shí)別鍵盤的Alt鍵,要加入其他的快捷鍵,必須自己實(shí)現(xiàn)一個(gè)ActionListener。

      d.通過setLayout(null)可以使得所有組件以setBounds()的四個(gè)參數(shù)來(lái)精確定位各自的大小、位置,但不推薦使用,因?yàn)楹玫木幊田L(fēng)格不應(yīng)該在Swing代碼中硬編碼具體數(shù)字,所有的數(shù)字應(yīng)該以常數(shù)的形式統(tǒng)一存在一個(gè)靜態(tài)無(wú)實(shí)例資源類文件中。這個(gè)靜態(tài)無(wú)實(shí)例類統(tǒng)一負(fù)責(zé)Swing界面的風(fēng)格,包括字體和顏色都應(yīng)該包括進(jìn)去。

      e.好的界面設(shè)計(jì)有一條Golden Rule: 用戶不用任何手冊(cè)通過少數(shù)嘗試就能學(xué)會(huì)使用軟件。所以盡量把按鈕以菜單的形式(不管是右鍵菜單還是窗體自帶頂部菜單)呈現(xiàn)給顧客,除非是頻繁點(diǎn)擊的按鈕才有必要直接呈現(xiàn)在界面中。

      其實(shí)Swing的功能是相當(dāng)強(qiáng)大的,只是現(xiàn)在應(yīng)用不廣泛,專門去研究大概是要花不少時(shí)間的。筆者在各網(wǎng)站論壇瀏覽關(guān)于Swing的技巧文章還是比較可信的,自己所學(xué)非常有限,各人體會(huì)對(duì)Swing各個(gè)組件的掌握就是一個(gè)實(shí)踐積累的過程。筆者只用到過以上這些,所以只能談?wù)劜糠窒敕ǎ€望大家見諒!

      Java雜談

      (六)這篇是筆者打算寫的J2se部分的最后一篇了,這篇結(jié)束之后,再寫J2ee部分,不知道是否還合適寫在這個(gè)版塊?大家可以給點(diǎn)意見,謝謝大家對(duì)小弟這么鼓勵(lì)一路寫完前六篇Java雜談的J2se部分。最后這篇打算談一談Java中的RMI機(jī)制和JVM沙箱安全框架。

      1. Java中的RMI機(jī)制

      RMI的全稱是遠(yuǎn)程方法調(diào)用,相信不少朋友都聽說過,基本的思路可以用一個(gè)經(jīng)典比方來(lái)解釋:A計(jì)算機(jī)想要計(jì)算一個(gè)兩個(gè)數(shù)的加法,但A自己做不了,于是叫另外一臺(tái)計(jì)算機(jī)B幫忙,B有計(jì)算加法的功能,A調(diào)用它就像調(diào)用這個(gè)功能是自己的一樣方便。這個(gè)就叫做遠(yuǎn)程方法調(diào)用了。

      遠(yuǎn)程方法調(diào)用是EJB實(shí)現(xiàn)的支柱,建立分布式應(yīng)用的核心思想。這個(gè)很好理解,再拿上面的計(jì)算加法例子,A只知道去call計(jì)算機(jī)B的方法,自己并沒有B的那些功能,所以A計(jì)算機(jī)端就無(wú)法看到B執(zhí)行這段功能的過程和代碼,因?yàn)榭炊伎床坏?,所以既沒有機(jī)會(huì)竊取也沒有機(jī)會(huì)去改動(dòng)方法代碼。EJB正式基于這樣的思想來(lái)完成它的任務(wù)的。當(dāng)簡(jiǎn)單的加法變成復(fù)雜的數(shù)據(jù)庫(kù)操作和電子商務(wù)交易應(yīng)用的時(shí)候,這樣的安全性和分布式應(yīng)用的便利性就表現(xiàn)出來(lái)優(yōu)勢(shì)了。

      好了,回到細(xì)節(jié)上,要如何實(shí)現(xiàn)遠(yuǎn)程方法調(diào)用呢?我希望大家學(xué)習(xí)任何技術(shù)的時(shí)候可以試著依賴自己的下意識(shí)判斷,只要你的想法是合理健壯的,那么很可能實(shí)際上它就是這么做的,畢竟真理都蘊(yùn)藏在平凡的生活細(xì)節(jié)中。這樣只要帶著一些薄弱的Java基礎(chǔ)來(lái)思考RMI,其實(shí)也可以想出個(gè)大概來(lái)。

      a)需要有一個(gè)服務(wù)器角色,它擁有真正的功能代碼方法。例如B,它提供加法服務(wù)

      b)如果想遠(yuǎn)程使用B的功能,需要知道B的IP地址

      c)如果想遠(yuǎn)程使用B的功能,還需要知道B中那個(gè)特定服務(wù)的名字

      我們很自然可以想到這些,雖然不完善,但已經(jīng)很接近正確的做法了。實(shí)際上RMI要得以實(shí)現(xiàn)還得意于Java一個(gè)很重要的特性,就是Java反射機(jī)制。我們需要知道服務(wù)的名字,但又必須隱藏實(shí)現(xiàn)的代碼,如何去做呢?答案就是:接口!

      舉個(gè)例子:

      public interface Person(){

      public void sayHello();}

      Public class PersonImplA implements Person{

      public PersonImplA(){}

      public void sayHello(){ System.out.println(―Hello!‖);}

      }

      Public class PersonImplB implements Person{

      public PersonImplB(){}

      public void sayHello(){ System.out.println(―Nice to meet you!‖);}

      }

      客戶端:Person p = Naming.lookup(―PersonService‖);

      p.sayHello();

      就這幾段代碼就包含了幾乎所有的實(shí)現(xiàn)技術(shù),大家相信么?客戶端請(qǐng)求一個(gè)say hello服務(wù),服務(wù)器運(yùn)行時(shí)接到這個(gè)請(qǐng)求,利用Java反射機(jī)制的Class.newInstance()返回一個(gè)對(duì)象,但客戶端不知道服務(wù)器返回的是ImplA還是ImplB,它接受用的參數(shù)簽名是Person,它知道實(shí)現(xiàn)了Person接口的對(duì)象一定有sayHello()方法,這就意味著客戶端并不知道服務(wù)器真正如何去實(shí)現(xiàn)的,但它通過了解Person接口明確了它要用的服務(wù)方法名字叫做sayHello()。

      如此類推,服務(wù)器只需要暴露自己的接口出來(lái)供客戶端,所有客戶端就可以自己選擇需要的服務(wù)。這就像餐館只要拿出自己的菜單出來(lái)讓客戶選擇,就可以在后臺(tái)廚房一道道的按需做出來(lái),它怎么做的通常是不讓客戶知道的?。ㄗ?zhèn)鞑俗V吧,^_^)

      最后一點(diǎn)是我調(diào)用lookup,查找一個(gè)叫PersonService名字的對(duì)象,服務(wù)器只要看到這個(gè)名字,在自己的目錄(相當(dāng)于電話簿)中找到對(duì)應(yīng)的對(duì)象名字提供服務(wù)就可以了,這個(gè)目錄就叫做JNDI(Java命名與目錄接口),相信大家也聽過的。

      有興趣的朋友不妨自己做個(gè)RMI的應(yīng)用,很多前輩的博客中有簡(jiǎn)單的例子。提示一下利用Jdk的bin目錄中rmi.exe和rmiregistry.exe兩個(gè)命令就可以自己建起一個(gè)服務(wù)器,提供遠(yuǎn)程服務(wù)。因?yàn)槔雍苋菀渍遥揖筒蛔约号e例子了!

      2. JVM沙箱&框架

      RMI羅唆得太多了,實(shí)在是盡力想把它說清楚,希望對(duì)大家有幫助。最后的最后,給大家簡(jiǎn)單講一下JVM框架,我們叫做Java沙箱。Java沙箱的基本組件如下:

      a)類裝載器結(jié)構(gòu)

      b)class文件檢驗(yàn)器

      c)內(nèi)置于Java虛擬機(jī)的安全特性

      d)安全管理器及Java API

      其中類裝載器在3個(gè)方面對(duì)Java沙箱起作用: a.它防止惡意代碼去干涉善意的代碼

      b.它守護(hù)了被信任的類庫(kù)邊界

      c.它將代碼歸入保護(hù)域,確定了代碼可以進(jìn)行哪些操作

      虛擬機(jī)為不同的類加載器載入的類提供不同的命名空間,命名空間由一系列唯一的名稱組成,每一個(gè)被裝載的類將有一個(gè)名字,這個(gè)命名空間是由Java虛擬機(jī)為每一個(gè)類裝載器維護(hù)的,它們互相之間甚至不可見。

      我們常說的包(package)是在Java虛擬機(jī)第2版的規(guī)范第一次出現(xiàn),正確定義是由同一個(gè)類裝載器裝載的、屬于同一個(gè)包、多個(gè)類型的集合。類裝載器采用的機(jī)制是雙親委派模式。具體的加載器框架我在Java雜談

      (一)中已經(jīng)解釋過了,當(dāng)時(shí)說最外層的加載器是AppClassLoader,其實(shí)算上網(wǎng)絡(luò)層的話AppClassLoader也可以作為parent,還有更外層的加載器URLClassLoader。為了防止惡意攻擊由URL加載進(jìn)來(lái)的類文件我們當(dāng)然需要分不同的訪問命名空間,并且制定最安全的加載次序,簡(jiǎn)單來(lái)說就是兩點(diǎn):

      a.從最內(nèi)層JVM自帶類加載器開始加載,外層惡意同名類得不到先加載而無(wú)法使用

      b.由于嚴(yán)格通過包來(lái)區(qū)分了訪問域,外層惡意的類通過內(nèi)置代碼也無(wú)法獲得權(quán)限訪問到內(nèi)層類,破壞代碼就自然無(wú)法生效。

      附:關(guān)于Java的平臺(tái)無(wú)關(guān)性,有一個(gè)例子可以很明顯的說明這個(gè)特性:

      一般來(lái)說,C或C++中的int占位寬度是根據(jù)目標(biāo)平臺(tái)的字長(zhǎng)來(lái)決定的,這就意味著針對(duì)不同的平臺(tái)編譯同一個(gè)C++程序在運(yùn)行時(shí)會(huì)有不同的行為。然而對(duì)于Java中的int都是32位的二進(jìn)制補(bǔ)碼標(biāo)識(shí)的有符號(hào)整數(shù),而float都是遵守IEEE 754浮點(diǎn)標(biāo)準(zhǔn)的32位浮點(diǎn)數(shù)。

      PS: 這個(gè)小弟最近也沒時(shí)間繼續(xù)研究下去了,只是想拋磚引玉的提供給大家一個(gè)初步認(rèn)識(shí)JVM的印象。有機(jī)會(huì)了解一下JVM的內(nèi)部結(jié)構(gòu)對(duì)今后做Java開發(fā)是很有好處的。

      Java雜談

      (七)--接口& 組件、容器

      終于又靜下來(lái)繼續(xù)寫這個(gè)主題的續(xù)篇,前六篇主要講了一些J2se方面的經(jīng)驗(yàn)和感受,眼下Java應(yīng)用范圍已經(jīng)被J2ee占據(jù)了相當(dāng)大的一塊領(lǐng)域,有些人甚至聲稱Java被J2ee所取代了。不知道大家如何來(lái)理解所謂的J2ee(Java2 Enterprise Edition),也就是Java企業(yè)級(jí)應(yīng)用?

      筆者的觀點(diǎn)是,技術(shù)的發(fā)展是順應(yīng)世界變化的趨勢(shì)的,從C/S過渡到B/S模式,從客戶端的角度考慮企業(yè)級(jí)應(yīng)用或者說電子商務(wù)領(lǐng)域不在關(guān)心客戶端維護(hù)問題,這個(gè)任務(wù)已經(jīng)交給了任何一臺(tái)PC都會(huì)有的瀏覽器去維護(hù);從服務(wù)器端的角度考慮,以往C/S中的TCP/IP協(xié)議實(shí)現(xiàn)載體ServerSocket被Web Server Container所取代,例如大家都很熟悉的Tomcat、JBoss、WebLogic等等??傊磺械霓D(zhuǎn)變都是為了使得Java技術(shù)能更好的為人類生產(chǎn)生活所服務(wù)。

      有人會(huì)問,直接去學(xué)J2ee跳過J2se行否?筆者是肯定不贊成的,實(shí)際上確實(shí)有人走這條路,但筆者自身體會(huì)是正是由于J2se的基礎(chǔ)很牢固,才會(huì)導(dǎo)致在J2ee學(xué)習(xí)的道路上順風(fēng)順?biāo)?,知識(shí)點(diǎn)上不會(huì)有什么迷惑的地方。舉個(gè)簡(jiǎn)單的例子吧:

      筆者曾經(jīng)跟大學(xué)同學(xué)討論下面這兩種寫法的區(qū)別:

      ArrayList list = new ArrayList();//筆者不說反對(duì),但至少不贊成List list = new ArrayList();//筆者支持

      曾經(jīng)筆者跟同學(xué)爭(zhēng)論了幾個(gè)小時(shí),他非說第一種寫法更科學(xué),第二種完全沒有必要。我無(wú)法完全說服他,但筆者認(rèn)為良好的習(xí)慣和意識(shí)是任何時(shí)候都應(yīng)該針對(duì)接口編程,以達(dá)到解耦合和可擴(kuò)展性的目的。下面就以接口開始進(jìn)入J2ee的世界吧:

      1.J2ee與接口

      每一個(gè)版本的J2ee都對(duì)應(yīng)著一個(gè)確定版本的JDK,J2ee1.4對(duì)應(yīng)Jdk1.4,現(xiàn)在比較新的是JDK5.0,自然也會(huì)有J2EE 5.0。其實(shí)筆者一直在用的是J2EE1.4,不過沒什么關(guān)系,大家可以下任何一個(gè)版本的J2ee api來(lái)稍微瀏覽一下。筆者想先聲明一個(gè)概念,J2ee也是源自Java,所以底層的操作依然調(diào)用到很多J2se的庫(kù),所以才建議大家先牢牢掌握J(rèn)2se的主流技術(shù)。

      J2ee api有一個(gè)特點(diǎn),大家比較熟悉的幾個(gè)包java.jms、javax.servlet.http、javax.ejb等都以interface居多,實(shí)現(xiàn)類較少。其實(shí)大家真正在用的時(shí)候百分之六十以上都在反復(fù)的查著javax.servlet.http這個(gè)包下面幾個(gè)實(shí)現(xiàn)類的api函數(shù),其他的包很少問津。筆者建議在學(xué)習(xí)一種技術(shù)之前,對(duì)整體的框架有一個(gè)了解是很有必要的,J2ee旨在通過interface的聲明來(lái)規(guī)范實(shí)現(xiàn)的行為,任何第三方的廠商想要提供自己品牌的實(shí)現(xiàn)前提也是遵循這些接口定義的規(guī)則。如果在從前J2se學(xué)習(xí)的道路上對(duì)接口的理解很好的話,這里的體會(huì)將是非常深刻的,舉個(gè)簡(jiǎn)單的例子:

      public interface Mp3{

      public void play();

      public void record();

      public void stop();

      }

      如果我定義這個(gè)簡(jiǎn)單的接口,發(fā)布出去,規(guī)定任何第三方的公司想推出自己的名字為Mp3的產(chǎn)品都必須實(shí)現(xiàn)這個(gè)接口,也就是至少提供接口中方法的具體實(shí)現(xiàn)。這個(gè)意義已經(jīng)遠(yuǎn)遠(yuǎn)不止是面向?qū)ο蟮亩鄳B(tài)了,只有廠商遵循J2ee的接口定義,世界上的J2ee程序員才能針對(duì)統(tǒng)一的接口進(jìn)行程序設(shè)計(jì),最終不用改變代碼只是因?yàn)槭褂昧瞬煌瑥S商的實(shí)現(xiàn)類而有不同的特性罷了,本質(zhì)上說,無(wú)論哪一種廠商實(shí)現(xiàn)都完成了職責(zé)范圍內(nèi)的工作。這個(gè)就是筆者想一直強(qiáng)調(diào)的,針對(duì)接口編程的思想。

      接口到底有什么好處呢?我們這樣設(shè)想,現(xiàn)在有AppleMp3、SonyMp3、SamsungMp3都實(shí)現(xiàn)了這個(gè)Mp3的接口,于是都有了play、record、stop這三個(gè)功能。我們將Mp3產(chǎn)品座位一個(gè)組件的時(shí)候就不需要知道它的具體實(shí)現(xiàn),只要看到接口定義知道這個(gè)對(duì)象有3個(gè)功能就可以使用了。那么類似下面這樣的業(yè)務(wù)就完全可以在任何時(shí)間從3個(gè)品牌擴(kuò)展到任意個(gè)品牌,開個(gè)玩笑的說,項(xiàng)目經(jīng)理高高在上的寫完10個(gè)接口里的方法聲明,然后就丟給手下的程序員去寫里面的細(xì)節(jié),由于接口已經(jīng)統(tǒng)一(即每個(gè)方法傳入和傳出的格式已經(jīng)統(tǒng)一),經(jīng)理只需關(guān)注全局的業(yè)務(wù)就可以天天端杯咖啡走來(lái)走去了,^_^:

      public Mp3 create();

      public void copy(Mp3 mp3);

      public Mp3 getMp3();

      最后用一個(gè)簡(jiǎn)單的例子說明接口:一個(gè)5號(hào)電池的手電筒,可以裝入任何牌子的5號(hào)電池,只要它符合5號(hào)電池的規(guī)范,裝入之后任何看不到是什么牌子,只能感受到手電筒在完成它的功能。那么生產(chǎn)手電筒的廠商和生產(chǎn)5號(hào)電池的廠商就可以完全解除依賴關(guān)系,可以各自自由開發(fā)自己的產(chǎn)品,因?yàn)樗鼈兌甲袷?號(hào)電池應(yīng)有的形狀、正負(fù)極位置等約定。這下大家能對(duì)接口多一點(diǎn)體會(huì)了么?

      2.組件和容器

      針對(duì)接口是筆者特意強(qiáng)調(diào)的J2ee學(xué)習(xí)之路必備的思想,另外一個(gè)就是比較常規(guī)的組件和容器的概念了。很多教材和專業(yè)網(wǎng)站都說J2EE的核心是一組規(guī)范與指南,強(qiáng)調(diào)J2ee的核心概念就是組件+容器,這確實(shí)是無(wú)可厚非的。隨著越來(lái)越多的J2ee框架出現(xiàn),相應(yīng)的每種框架都一般有與之對(duì)應(yīng)的容器。

      容器,是用來(lái)管理組件行為的一個(gè)集合工具,組件的行為包括與外部環(huán)境的交互、組件的生命周期、組件之間的合作依賴關(guān)系等等。J2ee包含的容器種類大約有Web容器、Application Client容器、EJB容器、Applet客戶端容器等。但在筆者看來(lái),現(xiàn)在容器的概念變得有點(diǎn)模糊了,大家耳熟能詳是那些功能強(qiáng)大的開源框架,比如Hibernate、Struts2、Spring、JSF等,其中Hibernate就基于JDBC的基礎(chǔ)封裝了對(duì)事務(wù)和會(huì)話的管理,大大方便了對(duì)數(shù)據(jù)庫(kù)操作的繁瑣代碼,從這個(gè)意義上來(lái)說它已經(jīng)接近容器的概念了,EJB的實(shí)體Bean也逐漸被以Hibernate為代表的持久化框架所取代。

      組件,本意是指可以重用的代碼單元,一般代表著一個(gè)或者一組可以獨(dú)立出來(lái)的功能模塊,在J2ee中組件的種類有很多種,比較常見的是EJB組件、DAO組件、客戶端組件或者應(yīng)用程序組件等,它們有個(gè)共同特點(diǎn)是分別會(huì)打包成.war,.jar,.jar,.ear,每個(gè)組件由特定格式的xml描述符文件進(jìn)行描述,而且服務(wù)器端的組件都需要被部署到應(yīng)用服務(wù)器上面才能夠被使用。

      稍微理解完組件和容器,還有一個(gè)重要的概念就是分層模型,最著名的當(dāng)然是MVC三層模型。在一個(gè)大的工程或項(xiàng)目中,為了讓前臺(tái)和后臺(tái)各個(gè)模塊的編程人員能夠同時(shí)進(jìn)行工作提高開發(fā)效率,最重要的就是實(shí)現(xiàn)層與層之間的耦合關(guān)系,許多分層模型的宗旨和開源框架所追求的也就是這樣的效果。在筆者看來(lái),一個(gè)完整的Web項(xiàng)目大概有以下幾個(gè)層次:

      a)表示層(Jsp、Html、javascript、Ajax、Flash等等技術(shù)對(duì)其支持)

      b)控制層(Struts、JSF、WebWork等等框架在基于Servlet的基礎(chǔ)上支持,負(fù)責(zé)把具體的請(qǐng)求數(shù)據(jù)(有時(shí)卸載重新裝載)導(dǎo)向適合處理它的模型層對(duì)象)

      c)模型層(筆者認(rèn)為目前最好的框架是Spring,實(shí)質(zhì)就是處理表示層經(jīng)由控制層轉(zhuǎn)發(fā)過來(lái)的數(shù)據(jù),包含著大量的業(yè)務(wù)邏輯)

      d)數(shù)據(jù)層(Hibernate、JDBC、EJB等,由模型層處理完了持久化到數(shù)據(jù)庫(kù)中)

      當(dāng)然,這僅僅是筆者個(gè)人的觀點(diǎn),僅僅是供大家學(xué)習(xí)做一個(gè)參考,如果要實(shí)現(xiàn)這些層之間的完全分離,那么一個(gè)大的工程,可以僅僅通過增加人手就來(lái)完成任務(wù)。雖然《人月神話》中已經(jīng)很明確的闡述了增加人手并不能是效率增加,很大程度上是因?yàn)楸舜俗龅墓ぷ饔许樞蛏系囊蕾囮P(guān)系或者說難度和工作量上的巨大差距。當(dāng)然理想狀態(tài)在真實(shí)世界中是不可能達(dá)到的,但我們永遠(yuǎn)應(yīng)該朝著這個(gè)方向去不斷努力。最開始所提倡的針對(duì)接口來(lái)編程,哪怕是小小的細(xì)節(jié),寫一條List list= = new ArrayList()語(yǔ)句也能體現(xiàn)著處處皆使用接口的思想在里面。Anyway,這只是個(gè)開篇,筆者會(huì)就自己用過的J2ee技術(shù)和框架再細(xì)化談一些經(jīng)驗(yàn)

      Java雜談

      (八)--Servlet/Jsp

      終于正式進(jìn)入J2ee的細(xì)節(jié)部分了,首當(dāng)其沖的當(dāng)然是Servlet和Jsp了,上篇曾經(jīng)提到過J2ee只是一個(gè)規(guī)范和指南,定義了一組必須要遵循的接口,核心概念是組件和容器。曾經(jīng)有的人問筆者Servlet的Class文件是哪里來(lái)的?他認(rèn)為是J2ee官方提供的,我舉了一個(gè)簡(jiǎn)單的反例:稍微檢查了一下Tomcat5.0里面的Servlet.jar文件和JBoss里面的Servlet.jar文件大小,很明顯是不一樣的,至少已經(jīng)說明了它們不是源自同根的吧。其實(shí)Servlet是由容器根據(jù)J2ee的接口定義自己來(lái)實(shí)現(xiàn)的,實(shí)現(xiàn)的方式當(dāng)然可以不同,只要都遵守J2ee規(guī)范和指南。

      上述只是一個(gè)常見的誤區(qū)罷了,告訴我們要編譯運(yùn)行Servlet,是要依賴于實(shí)現(xiàn)它的容器的,不然連jar文件都沒有,編譯都無(wú)法進(jìn)行。那么Jsp呢?Java Server Page的簡(jiǎn)稱,是為了開發(fā)動(dòng)態(tài)網(wǎng)頁(yè)而誕生的技術(shù),其本質(zhì)也是Jsp,在編寫完畢之后會(huì)在容器啟動(dòng)時(shí)經(jīng)過編譯成對(duì)應(yīng)的Servlet。只是我們利用Jsp的很多新特性,可以更加專注于前后臺(tái)的分離,早期Jsp做前臺(tái)是滿流行的,畢竟里面支持Html代碼,這讓前臺(tái)美工人員可以更有效率的去完成自己的工作。然后Jsp將請(qǐng)求轉(zhuǎn)發(fā)到后臺(tái)的Servlet,由Servlet處理業(yè)務(wù)邏輯,再轉(zhuǎn)發(fā)回另外一個(gè)Jsp在前臺(tái)顯示出來(lái)。這似乎已經(jīng)成為一種常用的模式,最初筆者學(xué)習(xí)J2ee的時(shí)候,大量時(shí)間也在編寫這樣的代碼。

      盡管現(xiàn)在做前臺(tái)的技術(shù)越來(lái)越多,例如Flash、Ajax等,已經(jīng)有很多人不再認(rèn)為Jsp重要了。筆者覺得Jsp帶來(lái)的不僅僅是前后端分離的設(shè)計(jì)理念,它的另外一項(xiàng)技術(shù)成就了我們今天用的很多框架,那就是Tag標(biāo)簽技術(shù)。所以與其說是在學(xué)習(xí)Jsp,不如更清醒的告訴自己在不斷的理解Tag標(biāo)簽的意義和本質(zhì)。

      1. Servlet以及Jsp的生命周期

      Servlet是Jsp的實(shí)質(zhì),盡管容器對(duì)它們的處理有所區(qū)別。Servlet有init()方法初始化,service()方法進(jìn)行Web服務(wù),destroy()方法進(jìn)行銷毀,從生到滅都由容器來(lái)掌握,所以這些方法除非你想自己來(lái)實(shí)現(xiàn)Servlet,否則是很少會(huì)接觸到的。正是由于很少接觸,才容易被廣大初學(xué)者所忽略,希望大家至少記住Servlet生命周期方法都是回調(diào)方法?;卣{(diào)這個(gè)概念簡(jiǎn)單來(lái)說就是把自己注入另外一個(gè)類中,由它來(lái)調(diào)用你的方法,所謂的另外一個(gè)類就是Web容器,它只認(rèn)識(shí)接口和接口的方法,注入進(jìn)來(lái)的是怎樣的對(duì)象不管,它只會(huì)根據(jù)所需調(diào)用這個(gè)對(duì)象在接口定義存在的那些方法。由容器來(lái)調(diào)用的Servlet對(duì)象的初始化、服務(wù)和銷毀方法,所以叫做回調(diào)。這個(gè)概念對(duì)學(xué)習(xí)其他J2ee技術(shù)相當(dāng)關(guān)鍵!

      那么Jsp呢?本事上是Servlet,還是有些區(qū)別的,它的生命周期是這樣的:

      a)一個(gè)客戶端的Request到達(dá)服務(wù)器->

      b)判斷是否第一次調(diào)用-> 是的話編譯Jsp成Servlet

      c)否的話再判斷此Jsp是否有改變-> 是的話也重新編譯Jsp成Servlet

      d)已經(jīng)編譯最近版本的Servlet裝載所需的其他Class

      e)發(fā)布Servlet,即調(diào)用它的Service()方法

      所以Jsp號(hào)稱的是第一次Load緩慢,以后都會(huì)很快的運(yùn)行。從它的生命的周期確實(shí)不難看出來(lái)這個(gè)特點(diǎn),客戶端的操作很少會(huì)改變Jsp的源碼,所以它不需要編譯第二次就一直可以為客戶端提供服務(wù)。這里稍微解釋一下Http的無(wú)狀態(tài)性,因?yàn)榘l(fā)現(xiàn)很多人誤解,Http的無(wú)狀態(tài)性是指每次一張頁(yè)面顯示出來(lái)了,與服務(wù)器的連接其實(shí)就已經(jīng)斷開了,當(dāng)再次有提交動(dòng)作的時(shí)候,才會(huì)再次與服務(wù)器進(jìn)行連接請(qǐng)求提供服務(wù)。當(dāng)然還有現(xiàn)在比較流行的是Ajax與服務(wù)器異步通過xml交互的技術(shù),在做前臺(tái)的領(lǐng)域潛力巨大,筆者不是Ajax的高手,這里無(wú)法為大家解釋。

      2. Tag標(biāo)簽的本質(zhì)

      筆者之前說了,Jsp本身初衷是使得Web應(yīng)用前后臺(tái)的開發(fā)可以脫離耦合分開有效的進(jìn)行,可惜這個(gè)理念的貢獻(xiàn)反倒不如它帶來(lái)的Tag技術(shù)對(duì)J2ee的貢獻(xiàn)要大。也許已經(jīng)有很多人開始使用Tag技術(shù)了卻并不了解它。所以才建議大家在學(xué)習(xí)J2ee開始的時(shí)候一定要認(rèn)真學(xué)習(xí)Jsp,其實(shí)最重要的就是明白標(biāo)簽的本質(zhì)。

      Html標(biāo)簽我們都很熟悉了,有 、、、,Jsp帶來(lái)的Tag標(biāo)簽遵循同樣的格式,或者說更嚴(yán)格的Xml格式規(guī)范,例如 <jsp:include>、<jsp:useBean>、<c:if>、<c:forEach> 等等。它們沒有什么神秘的地方,就其源頭也還是Java Class而已,Tag標(biāo)簽的實(shí)質(zhì)也就是一段Java代碼,或者說一個(gè)Class文件。當(dāng)配置文件設(shè)置好去哪里尋找這些Class的路徑后,容器負(fù)責(zé)將頁(yè)面中存在的標(biāo)簽對(duì)應(yīng)到相應(yīng)的Class上,執(zhí)行那段特定的Java代碼,如此而已。</p><p>說得明白一點(diǎn)的話還是舉幾個(gè)簡(jiǎn)單的例子說明一下吧:</p><p><jsp:include> 去哪里找執(zhí)行什么class呢?首先這是個(gè)jsp類庫(kù)的標(biāo)簽,當(dāng)然要去jsp類庫(kù)尋找相應(yīng)的class了,同樣它也是由Web容器來(lái)提供,例如Tomcat就應(yīng)該去安裝目錄的lib文件夾下面的jsp-api.jar里面找,有興趣的可以去找一找??!</p><p><c:forEach> 又去哪里找呢?這個(gè)是由Jsp2.0版本推薦的和核心標(biāo)記庫(kù)的內(nèi)容,例如 <c:if> 就對(duì)應(yīng)在頁(yè)面中做if判斷的功能的一斷Java代碼。它的class文件在jstl.jar這個(gè)類庫(kù)里面,往往還需要和一個(gè)standard.jar類庫(kù)一起導(dǎo)入,放在具體Web項(xiàng)目的WEB-INF的lib目錄下面就可以使用了。</p><p>順便羅唆一句,Web Project的目錄結(jié)構(gòu)是相對(duì)固定的,因?yàn)槿萜鲿?huì)按照固定的路徑去尋找它需要的配置文件和資源,這個(gè)任何一本J2ee入門書上都有,這里就不介紹了。了解Tag的本質(zhì)還要了解它的工作原理,所以大家去J2ee的API里找到并研究這個(gè)包:javax.servlet.jsp.tagext。它有一些接口,和一些實(shí)現(xiàn)類,專門用語(yǔ)開發(fā)Tag,只有自己親自寫出幾個(gè)不同功能的標(biāo)簽,才算是真正理解了標(biāo)簽的原理。別忘記了自己開發(fā)的標(biāo)簽要自己去完成配置文件,容器只是集成了去哪里尋找jsp標(biāo)簽對(duì)應(yīng)class的路徑,自己寫的標(biāo)簽庫(kù)當(dāng)然要告訴容器去哪里找啦。</p><p>說了這么多,我們?yōu)槭裁匆脴?biāo)簽?zāi)兀客耆贘sp里面來(lái)個(gè) <% %> 就可以在里面任意寫Java代碼了,但是長(zhǎng)期實(shí)踐發(fā)現(xiàn)頁(yè)面代碼統(tǒng)一都是與html同風(fēng)格的標(biāo)記語(yǔ)言更加有助于美工人員進(jìn)行開發(fā)前臺(tái),它不需要懂Java,只要Java程序員給個(gè)列表告訴美工什么標(biāo)簽可以完成什么邏輯功能,他就可以專注于美工,也算是進(jìn)一步隔離了前后臺(tái)的工作吧!</p><p>3. 成就Web框架</p><p>框架是什么?曾經(jīng)看過這樣的定義:與模式類似,框架也是解決特定問題的可重用方法,框架是一個(gè)描述性的構(gòu)建塊和服務(wù)集合,開發(fā)人員可以用來(lái)達(dá)成某個(gè)目標(biāo)。一般來(lái)說,框架提供了解決某類問題的基礎(chǔ)設(shè)施,是用來(lái)創(chuàng)建解決方案的工具,而不是問題的解決方案。</p><p>正是由于Tag的出現(xiàn),成就了以后出現(xiàn)的那么多Web框架,它們都開發(fā)了自己成熟實(shí)用的一套標(biāo)簽,然后由特定的Xml文件來(lái)配置加載信息,力圖使得Web應(yīng)用的開發(fā)變得更加高效。下面這些標(biāo)簽相應(yīng)對(duì)很多人來(lái)說相當(dāng)熟悉了:</p><p><html:password></p><p><logic:equal></p><p><bean:write></p><p><f:view></p><p><h:form></p><p><h:message></p><p>它們分別來(lái)自Struts和JSF框架,最強(qiáng)大的功能在于控制轉(zhuǎn)發(fā),就是MVC三層模型中間完成控制器的工作。Struts-1實(shí)際上并未做到真正的三層隔離,這一點(diǎn)在Struts-2上得到了很大的改進(jìn)。而Jsf向來(lái)以比較完善合理的標(biāo)簽庫(kù)受到人們推崇。</p><p>今天就大概講這么多吧,再次需要強(qiáng)調(diào)的是Servlet/Jsp是學(xué)習(xí)J2ee必經(jīng)之路,也是最基礎(chǔ)的知識(shí),希望大家給與足夠的重視!</p><p>Java雜談</p><p>(九)--Struts</p><p>J2ee的開源框架很多,筆者只能介紹自己熟悉的幾個(gè),其他的目前在中國(guó)IT行業(yè)應(yīng)用得不是很多。希望大家對(duì)新出的框架不要盲目的推崇,首先一定要熟悉它比舊的到底好在哪里,新的理念和特性是什么?然后再?zèng)Q定是否要使用它。</p><p>這期的主題是Struts,直譯過來(lái)是支架。Struts的第一個(gè)版本是在2001年5月發(fā)布的,它提供了一個(gè)Web應(yīng)用的解決方案,如何讓Jsp和servlet共存去提供清晰的分離視圖和業(yè)務(wù)應(yīng)用邏輯的架構(gòu)。在Struts之前,通常的做法是在Jsp中加入業(yè)務(wù)邏輯,或者在Servlet中生成視圖轉(zhuǎn)發(fā)到前臺(tái)去。Struts帶著MVC的新理念當(dāng)時(shí)退出幾乎成為業(yè)界公認(rèn)的Web應(yīng)用標(biāo)準(zhǔn),于是當(dāng)代IT市場(chǎng)上也出現(xiàn)了眾多熟悉Struts的程序員。即使有新的框架再出來(lái)不用,而繼續(xù)用Struts的理由也加上了一條低風(fēng)險(xiǎn),因?yàn)橹型救绻_發(fā)人員變動(dòng),很容易的招進(jìn)新的會(huì)Struts的IT民工啊,^_^!</p><p>筆者之前說的都是Struts-1,因?yàn)樾鲁隽薙truts-2,使得每次談到Struts都必須注明它是Struts-1還是2。筆者先談比較熟悉的Struts-1,下次再介紹一下與Struts-2的區(qū)別:</p><p>1. Struts框架整體結(jié)構(gòu)</p><p>Struts-1的核心功能是前端控制器,程序員需要關(guān)注的是后端控制器。前端控制器是是一個(gè)Servlet,在Web.xml中間配置所有Request都必須經(jīng)過前端控制器,它的名字是ActionServlet,由框架來(lái)實(shí)現(xiàn)和管理。所有的視圖和業(yè)務(wù)邏輯隔離都是應(yīng)為這個(gè)ActionServlet,它就像一個(gè)交通警察,所有過往的車輛必須經(jīng)過它的法眼,然后被送往特定的通道。所有,對(duì)它的理解就是分發(fā)器,我們也可以叫做Dispatcher,其實(shí)了解Servlet編程的人自己也可以寫一個(gè)分發(fā)器,加上攔截request的Filter,其實(shí)自己實(shí)現(xiàn)一個(gè)struts框架并不是很困難。主要目的就是讓編寫視圖的和后臺(tái)邏輯的可以脫離緊耦合,各自同步的完成自己的工作。</p><p>那么有了ActionServlet在中間負(fù)責(zé)轉(zhuǎn)發(fā),前端的視圖比如說是Jsp,只需要把所有的數(shù)據(jù)Submit,這些數(shù)據(jù)就會(huì)到達(dá)適合處理它的后端控制器Action,然后在里面進(jìn)行處理,處理完畢之后轉(zhuǎn)發(fā)到前臺(tái)的同一個(gè)或者不同的視圖Jsp中間,返回前臺(tái)利用的也是Servlet里面的forward和redirect兩種方式。所以到目前為止,一切都只是借用了Servlet的API搭建起了一個(gè)方便的框架而已。這也是Struts最顯著的特性——控制器。</p><p>那么另外一個(gè)特性,可以說也是Struts-1帶來(lái)的一個(gè)比較成功的理念,就是以xml配置代替硬編碼配置信息。以往決定Jsp往哪個(gè)servlet提交,是要寫進(jìn)Jsp代碼中的,也就是說一旦這個(gè)提交路徑要改,我們必須改寫代碼再重新編譯。而Struts提出來(lái)的思路是,編碼的只是一個(gè)邏輯名字,它對(duì)應(yīng)哪個(gè)class文件寫進(jìn)了xml配置文件中,這個(gè)配置文件記錄著所有的映射關(guān)系,一旦需要改變路徑,改變xml文件比改變代碼要容易得多。這個(gè)理念可以說相當(dāng)成功,以致于后來(lái)的框架都延續(xù)著這個(gè)思路,xml所起的作用也越來(lái)越大。</p><p>大致上來(lái)說Struts當(dāng)初給我們帶來(lái)的新鮮感就這么多了,其他的所有特性都是基于方便的控制轉(zhuǎn)發(fā)和可擴(kuò)展的xml配置的基礎(chǔ)之上來(lái)完成它們的功能的。</p><p>下面將分別介紹Action和FormBean,這兩個(gè)是Struts中最核心的兩個(gè)組件。</p><p>2. 后端控制器Action</p><p>Action就是我們說的后端控制器,它必須繼承自一個(gè)Action父類,Struts設(shè)計(jì)了很多種Action,例如DispatchAction、DynaValidationAction。它們都有一個(gè)處理業(yè)務(wù)邏輯的方法execute(),傳入的request, response, formBean和actionMapping四個(gè)對(duì)象,返回actionForward對(duì)象。到達(dá)Action之前先會(huì)經(jīng)過一個(gè)RequestProcessor來(lái)初始化配置文件的映射關(guān)系,這里需要大家注意幾點(diǎn):</p><p>1)為了確保線程安全,在一個(gè)應(yīng)用的生命周期中,Struts框架只會(huì)為每個(gè)Action類創(chuàng)建一個(gè)Action實(shí)例,所有的客戶請(qǐng)求共享同一個(gè)Action實(shí)例,并且所有線程可以同時(shí)執(zhí)行它的execute()方法。所以當(dāng)你繼承父類Action,并添加了private成員變量的時(shí)候,請(qǐng)記住這個(gè)變量可以被多個(gè)線程訪問,它的同步必須由程序員負(fù)責(zé)。(所有我們不推薦這樣做)。在使用Action的時(shí)候,保證線程安全的重要原則是在Action類中僅僅使用局部變量,謹(jǐn)慎的使用實(shí)例變量。局部變量是對(duì)每個(gè)線程來(lái)說私有的,execute方法結(jié)束就被銷毀,而實(shí)例變量相當(dāng)于被所有線程共享。</p><p>2)當(dāng)ActionServlet實(shí)例接收到Http請(qǐng)求后,在doGet()或者doPost()方法中都會(huì)調(diào)用process()方法來(lái)處理請(qǐng)求。RequestProcessor類包含一個(gè)HashMap,作為存放所有Action實(shí)例的緩存,每個(gè)Action實(shí)例在緩存中存放的屬性key為Action類名。在RequestProcessor類的processActionCreate()方法中,首先檢查在HashMap中是否存在Action實(shí)例。創(chuàng)建Action實(shí)例的代碼位于同步代碼塊中,以保證只有一個(gè)線程創(chuàng)建Action實(shí)例。一旦線程創(chuàng)建了Action實(shí)例并把它存放到HashMap中,以后所有的線程會(huì)直接使用這個(gè)緩存中的實(shí)例。</p><p>3)<action> 元素的 <roles> 屬性指定訪問這個(gè)Action用戶必須具備的安全角色,多個(gè)角色之間逗號(hào)隔開。RequestProcessor類在預(yù)處理請(qǐng)求時(shí)會(huì)調(diào)用自身的processRoles()方法,檢查配置文件中是否為Action配置了安全角色,如果有,就調(diào)用HttpServletRequest的isUserInRole()方法來(lái)判斷用戶是否具備了必要的安全性角色,如果不具備,就直接向客戶端返回錯(cuò)誤。(返回的視圖通過 <input> 屬性來(lái)指定)</p><p>3. 數(shù)據(jù)傳輸對(duì)象FormBean</p><p>Struts并沒有把模型層的業(yè)務(wù)對(duì)象直接傳遞到視圖層,而是采用DTO(Data Transfer Object)來(lái)傳輸數(shù)據(jù),這樣可以減少傳輸數(shù)據(jù)的冗余,提高傳輸效率;還有助于實(shí)現(xiàn)各層之間的獨(dú)立,使每個(gè)層分工明確。Struts的DTO就是ActionForm,即formBean。由于模型層應(yīng)該和Web應(yīng)用層保持獨(dú)立。由于ActionForm類中使用了Servlet API,因此不提倡把ActionForm傳遞給模型層,而應(yīng)該在控制層把ActionForm Bean的數(shù)據(jù)重新組裝到自定義的DTO中,再把它傳遞給模型層。它只有兩個(gè)scope,分別是session和request。(默認(rèn)是session)一個(gè)ActionForm標(biāo)準(zhǔn)的生命周期是:</p><p>1)控制器收到請(qǐng)求-></p><p>2)從request或session中取出ActionForm實(shí)例,如不存在就創(chuàng)建一個(gè)-> 3)調(diào)用ActionForm的reset()方法-> 4)把實(shí)例放入session或者request中-> 5)將用戶輸入表達(dá)數(shù)據(jù)組裝到ActionForm中-> 6)如眼張方法配置了就調(diào)用validate()方法-></p><p>7)如驗(yàn)證錯(cuò)誤就轉(zhuǎn)發(fā)給 <input> 屬性指定的地方,否則調(diào)用execute()方法</p><p>validate()方法調(diào)用必須滿足兩個(gè)條件:</p><p>1)ActionForm 配置了Action映射而且name屬性匹配</p><p>2)<aciton> 元素的validate屬性為true</p><p>如果ActionForm在request范圍內(nèi),那么對(duì)于每個(gè)新的請(qǐng)求都會(huì)創(chuàng)建新的ActionForm實(shí)例,屬性被初始化為默認(rèn)值,那么reset()方法就顯得沒有必要;但如果ActionForm在session范圍內(nèi),同一個(gè)ActionForm實(shí)例會(huì)被多個(gè)請(qǐng)求共享,reset()方法在這種情況下極為有用。</p><p>4. 驗(yàn)證框架和國(guó)際化</p><p>Struts有許多自己的特性,但是基本上大家還是不太常用,說白了它們也是基于JDK中間的很多Java基礎(chǔ)包來(lái)完成工作。例如國(guó)際化、驗(yàn)證框架、插件自擴(kuò)展功能、與其他框架的集成、因?yàn)楦鞔罂蚣芑径加刑峁┻@樣的特性,Struts也并不是做得最好的一個(gè),這里也不想多說。Struts的驗(yàn)證框架,是通過一個(gè)validator.xml的配置文件讀入驗(yàn)證規(guī)則,然后在validation-rules.xml里面找到驗(yàn)證實(shí)現(xiàn)通過自動(dòng)為Jsp插入javascript來(lái)實(shí)現(xiàn),可以說做得相當(dāng)簡(jiǎn)陋。彈出來(lái)的javascript框不但難看還很多冗余信息,筆者寧愿用formBean驗(yàn)證或者Action的saveErrors(),驗(yàn)證邏輯雖然要自己寫,但頁(yè)面隱藏/浮現(xiàn)的警告提示更加人性化和美觀一些。</p><p>至于Struts的國(guó)際化,其實(shí)無(wú)論哪個(gè)框架的國(guó)際化,java.util.Locale類是最重要的Java I18N類。在Java語(yǔ)言中,幾乎所有的對(duì)國(guó)際化和本地化的支持都依賴于這個(gè)類。如果Java類庫(kù)中的某個(gè)類在運(yùn)行的時(shí)候需要根據(jù)Locale對(duì)象來(lái)調(diào)整其功能,那么就稱這個(gè)類是本地敏感的(Locale-Sensitive),例如java.text.DateFormat類就是,依賴于特定Locale。</p><p>創(chuàng)建Locale對(duì)象的時(shí)候,需要明確的指定其語(yǔ)言和國(guó)家的代碼,語(yǔ)言代碼遵從的是ISO-639規(guī)范,國(guó)家代碼遵從ISO-3166規(guī)范,可以從</p><p>http:// http://。</p><p>這里很明顯的一點(diǎn)是不存在FormBean的作用域封裝,直接可以從Action中取得數(shù)據(jù)。這里有一個(gè)Strut-2配置的web.xml文件:</p><p><filter></p><p><filter-name> controller </filter-name></p><p><filter-class> org.apache.struts.action2.dispatcher.FilterDispatcher </filter-class></p><p></filter></p><p><filter-mapping></p><p><filter-name> cotroller </filter-name></p><p><url-pattern> /* </url-pattern></p><p></filter-mapping></p><p>注意到以往的servlet變成了filter,ActionServlet變成了FilterDispatcher,*.do變成了/*。filter配置定義了名稱(供關(guān)聯(lián))和filter的類。filter mapping讓URI匹配成功的的請(qǐng)求調(diào)用該filter。默認(rèn)情況下,擴(kuò)展名為 “.action ”。這個(gè)是在default.properties文件里的 “struts.action.extension ”屬性定義的。</p><p>default.properties是屬性定義文件,通過在項(xiàng)目classpath路徑中包含一個(gè)名為―struts.properties‖的文件來(lái)設(shè)置不同的屬性值。而Struts-2的默認(rèn)配置文件名為struts.xml。由于1和2的action擴(kuò)展名分別為.do和.action,所以很方便能共存。我們?cè)賮?lái)看一個(gè)Struts-2的action代碼:</p><p>public class MyAction {</p><p>public String execute()throws Exception {</p><p>//do the work return “success ”;</p><p>}</p><p>}</p><p>很明顯的區(qū)別是不用再繼承任何類和接口,返回的只是一個(gè)String,無(wú)參數(shù)。實(shí)際上在Struts-2中任何返回String的無(wú)參數(shù)方法都可以通過配置來(lái)調(diào)用action。所有的參數(shù)從哪里來(lái)獲得呢?答案就是Inversion of Control技術(shù)(控制反轉(zhuǎn))。筆者盡量以最通俗的方式來(lái)解釋,我們先試圖讓這個(gè)Action獲得reuqest對(duì)象,這樣可以提取頁(yè)面提交的任何參數(shù)。那么我們把request設(shè)為一個(gè)成員變量,然后需要一個(gè)對(duì)它的set方法。由于大部分的action都需要這么做,我們把這個(gè)set方法作為接口來(lái)實(shí)現(xiàn)。</p><p>public interface ServletRequestAware {</p><p>public void setServletRequest(HttpServletRequest request);</p><p>}</p><p>public class MyAction implements ServletRequestAware {</p><p>private HttpServletRequest request;</p><p>public void setServletRequest(HttpServletRequest request){</p><p>this.request = request;</p><p>}</p><p>public String execute()throws Exception {</p><p>// do the work directly using the request</p><p>return Action.SUCCESS;</p><p>}</p><p>}</p><p>那么誰(shuí)來(lái)調(diào)用這個(gè)set方法呢?也就是說誰(shuí)來(lái)控制這個(gè)action的行為,以往我們都是自己在適當(dāng)?shù)牡胤綄懮弦痪鋋ction.setServletRequest(…),也就是控制權(quán)在程序員這邊。然而控制反轉(zhuǎn)的思想是在哪里調(diào)用交給正在運(yùn)行的容器來(lái)決定,只要利用Java反射機(jī)制來(lái)獲得Method對(duì)象然后調(diào)用它的invoke方法傳入?yún)?shù)就能做到,這樣控制權(quán)就從程序員這邊轉(zhuǎn)移到了容器那邊。程序員可以減輕很多繁瑣的工作更多的關(guān)注業(yè)務(wù)邏輯。Request可以這樣注入到action中,其他任何對(duì)象也都可以。為了保證action的成員變量線程安全,Struts-2的action不是單例的,每一個(gè)新的請(qǐng)求都會(huì)產(chǎn)生一個(gè)新的action實(shí)例。</p><p>那么有人會(huì)問,到底誰(shuí)來(lái)做這個(gè)對(duì)象的注入工作呢?答案就是攔截器。攔截器又是什么東西?筆者再來(lái)盡量通俗的解釋攔截器的概念。大家要理解攔截器的話,首先一定要理解GOF23種設(shè)計(jì)模式中的Proxy模式。</p><p>A對(duì)象要調(diào)用f(),它希望代理給B來(lái)做,那么B就要獲得A對(duì)象的引用,然后在B的f()中通過A對(duì)象引用調(diào)用A對(duì)象的f()方法,最終達(dá)到A的f()被調(diào)用的目的。有沒有人會(huì)覺得這樣很麻煩,為什么明明只要A.f()就可以完成的一定要封裝到B的f()方法中去?有哪些好處呢?</p><p>1)這里我們只有一個(gè)A,當(dāng)我們有很多個(gè)A的時(shí)候,只需要監(jiān)視B一個(gè)對(duì)象的f()方法就可以從全局上控制所有被調(diào)用的f()方法。</p><p>2)另外,既然代理人B能獲得A對(duì)象的引用,那么B可以決定在真正調(diào)A對(duì)象的f()方法之前可以做哪些前置工作,調(diào)完返回前可有做哪些后置工作。</p><p>講到這里,大家看出來(lái)一點(diǎn)攔截器的概念了么?它攔截下一調(diào)f()方法的請(qǐng)求,然后統(tǒng)一的做處理(處理每個(gè)的方式還可以不同,解析A對(duì)象就可以辨別),處理完畢再放行。這樣像不像對(duì)流動(dòng)的河水橫切了一刀,對(duì)所有想通過的水分子進(jìn)行搜身,然后再放行?這也就是AOP(Aspect of Programming面向切面編程)的思想。</p><p>Anyway,Struts-2只是利用了AOP和IoC技術(shù)來(lái)減輕action和框架的耦合關(guān)系,力圖到最大程度重用action的目的。在這樣的技術(shù)促動(dòng)下,Struts-2的action成了一個(gè)簡(jiǎn)單被框架使用的POJO(Plain Old Java Object)罷了。實(shí)事上AOP和IoC的思想已經(jīng)遍布新出來(lái)的每一個(gè)框架上,他們并不是多么新的技術(shù),利用的也都是JDK早已可以最到的事情,它們代表的是更加面向接口編程,提高重用,增加擴(kuò)展性的一種思想。Struts-2只是部分的使用這兩種思想來(lái)設(shè)計(jì)完成的,另外一個(gè)最近很火的框架Spring,更大程度上代表了這兩種設(shè)計(jì)思想,筆者將于下一篇來(lái)進(jìn)一步探討Spring的結(jié)構(gòu)。</p><p>PS: 關(guān)于Struts-2筆者也沒真正怎么用過,這里是看了網(wǎng)上一些前輩的帖子之后寫下自己的學(xué)習(xí)體驗(yàn),不足之處請(qǐng)見諒!</p><p>Java雜談</p><p>(十一)--Spring</p><p>筆者最近比較忙,一邊在實(shí)習(xí)一邊在尋找明年畢業(yè)更好的工作,不過論壇里的朋友非常支持小弟繼續(xù)寫,今天是周末,泡上一杯咖啡,繼續(xù)與大家分享J2ee部分的學(xué)習(xí)經(jīng)驗(yàn)。今天的主題是目前很流行也很好的一個(gè)開源框架-Spring。</p><p>引用《Spring2.0技術(shù)手冊(cè)》上的一段話:</p><p>Spring的核心是個(gè)輕量級(jí)容器,它是實(shí)現(xiàn)IoC容器和非侵入性的框架,并提供AOP概念的實(shí)現(xiàn)方式;提供對(duì)持久層、事務(wù)的支持;提供MVC Web框架的實(shí)現(xiàn),并對(duì)于一些常用的企業(yè)服務(wù)API提供一致的模型封裝,是一個(gè)全方位的應(yīng)用程序框架,除此之外,對(duì)于現(xiàn)存的各種框架,Spring也提供了與它們相整合的方案。</p><p>接下來(lái)筆者先談?wù)勛约旱囊恍├斫獍?,Spring框架的發(fā)起者之前一本很著名的書名字大概是《J2ee Development without EJB》,他提倡用輕量級(jí)的組件代替重量級(jí)的EJB。筆者還沒有看完那本著作,只閱讀了部分章節(jié)。其中有一點(diǎn)分析覺得是很有道理的:</p><p>EJB里在服務(wù)器端有Web Container和EJB Container,從前的觀點(diǎn)是各層之間應(yīng)該在物理上隔離,Web Container處理視圖功能、在EJB Container中處理業(yè)務(wù)邏輯功能、然后也是EBJ Container控制數(shù)據(jù)庫(kù)持久化。這樣的層次是很清晰,但是一個(gè)很嚴(yán)重的問題是Web Container和EJB Container畢竟是兩個(gè)不同的容器,它們之間要通信就得用的是RMI機(jī)制和JNDI服務(wù),同樣都在服務(wù)端,卻物理上隔離,而且每次業(yè)務(wù)請(qǐng)求都要遠(yuǎn)程調(diào)用,有沒有必要呢?看來(lái)并非隔離都是好的。</p><p>再看看輕量級(jí)和重量級(jí)的區(qū)別,筆者看過很多種說法,覺得最有道理的是輕量級(jí)代表是POJO + IoC,重量級(jí)的代表是Container + Factory。(EJB2.0是典型的重量級(jí)組件的技術(shù))我們盡量使用輕量級(jí)的Pojo很好理解,意義就在于兼容性和可適應(yīng)性,移植不需要改變?cè)瓉?lái)的代碼。而Ioc與Factory比起來(lái),Ioc的優(yōu)點(diǎn)是更大的靈活性,通過配置可以控制很多注入的細(xì)節(jié),而Factory模式,行為是相對(duì)比較封閉固定的,生產(chǎn)一個(gè)對(duì)象就必須接受它全部的特點(diǎn),不管是否需要。其實(shí)輕量級(jí)和重量級(jí)都是相對(duì)的概念,使用資源更少、運(yùn)行負(fù)載更小的自然就算輕量。</p><p>話題扯遠(yuǎn)了,因?yàn)镾pring框架帶來(lái)了太多可以探討的地方。比如它的非侵入性:指的是它提供的框架實(shí)現(xiàn)可以讓程序員編程卻感覺不到框架的存在,這樣所寫的代碼并沒有和框架綁定在一起,可以隨時(shí)抽離出來(lái),這也是Spring設(shè)計(jì)的目標(biāo)。Spring是唯一可以做到真正的針對(duì)接口編程,處處都是接口,不依賴綁定任何實(shí)現(xiàn)類。同時(shí),Spring還設(shè)計(jì)了自己的事務(wù)管理、對(duì)象管理和Model2 的MVC框架,還封裝了其他J2ee的服務(wù)在里面,在實(shí)現(xiàn)上基本都在使用依賴注入和AOP的思想。由此我們大概可以看到Spring是一個(gè)什么概念上的框架,代表了很多優(yōu)秀思想,值得深入學(xué)習(xí)。筆者強(qiáng)調(diào),學(xué)習(xí)并不是框架,而是框架代表的思想,就像我們當(dāng)初學(xué)Struts一樣……</p><p>1.Spring MVC</p><p>關(guān)于IoC和AOP筆者在上篇已經(jīng)稍微解釋過了,這里先通過Spring的MVC框架來(lái)給大家探討一下Spring的特點(diǎn)吧。(畢竟大部分人已經(jīng)很熟悉Struts了,對(duì)比一下吧)</p><p>眾所周知MVC的核心是控制器。類似Struts中的ActionServlet,Spring里面前端控制器叫做DispatcherServlet。里面充當(dāng)Action的組件叫做Controller,返回的視圖層對(duì)象叫做ModelAndView,提交和返回都可能要經(jīng)過過濾的組件叫做Interceptor。</p><p>讓我們看看一個(gè)從請(qǐng)求到返回的流程吧:</p><p>(1)前臺(tái)Jsp或Html通過點(diǎn)擊submit,將數(shù)據(jù)裝入了request域</p><p>(2)請(qǐng)求被Interceptor攔截下來(lái),執(zhí)行preHandler()方法出前置判斷</p><p>(3)請(qǐng)求到達(dá)DispathcerServlet</p><p>(4)DispathcerServlet通過Handler Mapping來(lái)決定每個(gè)reuqest應(yīng)該轉(zhuǎn)發(fā)給哪個(gè)后端控制器Controller</p><p>(5)各式各樣的后端控制器Controller來(lái)處理請(qǐng)求,調(diào)用業(yè)務(wù)層對(duì)象來(lái)處理業(yè)務(wù)邏輯,然后返回一個(gè)ModelAndView對(duì)象</p><p>(6)當(dāng)Controller執(zhí)行完畢,Interceptor會(huì)調(diào)用postHandle來(lái)做后置處理</p><p>(7)ModelAndView代表了呈現(xiàn)畫面是使用的Model數(shù)據(jù)對(duì)象和View對(duì)象,由于只能返回一個(gè)對(duì)象所有起了這個(gè)名字封裝這兩個(gè)對(duì)象。</p><p>(8)由ViewResolver對(duì)象來(lái)解析每個(gè)返回的ModelAndView對(duì)象應(yīng)該呈現(xiàn)到哪一個(gè)視圖(Jsp/Html等)中(包括Exception Resolver)</p><p>(9)當(dāng)View繪制完成之后Interceptor又會(huì)跳出來(lái)執(zhí)行它的afterCompletion方法做善后處理。當(dāng)然Interceptor的行為完全是配置的而不是強(qiáng)制的。</p><p>這樣一個(gè)完整的流程就這樣結(jié)束了,個(gè)人感覺Spring的MVC框架稍顯復(fù)雜,不像Struts-1那么容易上手。不管是Controller、Model、ViewRosovler、Handle Mapping還是View,Spring MVC框架都已經(jīng)為你提供了多種實(shí)現(xiàn),想最大程度的減少程序員的編碼,增加框架的適用性。大家有興趣可以繼續(xù)深入研究哈!</p><p>2.Spring AOP</p><p>記得最初筆者請(qǐng)教他人Spring是一個(gè)什么東西的時(shí)候,每個(gè)人都會(huì)提到AOP這個(gè)詞語(yǔ)。筆者在上一篇已經(jīng)解釋過AOP基本原理,這次來(lái)跟大家說說Spring的AOP編程吧。不同的AOP框架會(huì)有其對(duì)AOP概念不同的實(shí)現(xiàn)方式,主要的差別在于所提供的Pointcut、Aspects的豐富程度,以及它們?nèi)绾伪豢椚霊?yīng)用程序、代理的方式等等。先熟悉一下AOP中的幾個(gè)重要概念:</p><p>(1)Cross-cutting:橫切,說白了就是需要統(tǒng)一處理的集合(2)Aspects:將散落各處的橫切收集起來(lái),設(shè)計(jì)成各個(gè)獨(dú)立可重用的對(duì)象稱為Aspects。</p><p>(3)Advice: 對(duì)橫切的具體實(shí)現(xiàn),即等待插入一段邏輯。</p><p>(4)Joinpoint:Advice插入流程的時(shí)機(jī)點(diǎn)。</p><p>(5)Pointcut: 用于選擇Joinpoint的程序結(jié)構(gòu),可以通過Annotation或者XML實(shí)現(xiàn)。</p><p>(6)Weave: Advice被應(yīng)用至對(duì)象之上的過程稱之為織入,有編譯期、類加載期、運(yùn)行期三種時(shí)間點(diǎn)策略。</p><p>如果你采用實(shí)現(xiàn)接口的方式,Spring會(huì)在執(zhí)行時(shí)期適用java的動(dòng)態(tài)代理,如果不實(shí)現(xiàn)接口,Spring會(huì)使用CGLIB產(chǎn)生代理類。AOP的概念很大很泛,而Spring只使用了其中的部分特性,畢竟Spring的目標(biāo)是輕量級(jí)框架,比如它只支持對(duì)Method的Joinpoint,而不支持對(duì)Field的Joinpoint,理由是為了封裝性。</p><p>其實(shí)我們可以把概念看得簡(jiǎn)單一點(diǎn),AOP的目的是減少冗余代碼,增強(qiáng)對(duì)較大項(xiàng)目的全局監(jiān)控。Spring利用AOP可以規(guī)定一個(gè)集合和一套規(guī)則,在這個(gè)集合里所有的方法被invoke即調(diào)用的時(shí)候,都必須按照那套規(guī)則走一遍。那么首先對(duì)其中10個(gè)方法都要用到的處理代碼就只用寫一遍,如果是這10個(gè)方法來(lái)了就織入這段代碼;其次,按照規(guī)則,也許所有的牽扯某個(gè)模塊的方法調(diào)用的時(shí)候,我都需要做日志或者進(jìn)行驗(yàn)證,那么我只要立足于這個(gè)集合的入口和出口,管他從哪里來(lái)去哪里,都能被有效的監(jiān)控。我監(jiān)控的可能不止是某個(gè)方法單獨(dú)的行為,我還可以加入對(duì)流程控制的監(jiān)控規(guī)則。例如是論壇,我規(guī)定注冊(cè)了才能登錄,而登錄后才能發(fā)帖回帖下資源,于是所有這類流程都會(huì)被收集到我眼皮地下通過。</p><p>PS:筆者最近忙于找工作的事,沒有太多經(jīng)歷在論壇跟大家整理自己的筆記。最近也只是接觸Spring的MVC比較多,對(duì)于Spring的其他特性,還沒有更多的去實(shí)踐,所以僅僅是泛泛而談,只是介紹一個(gè)印象罷了。還是那句話,我們學(xué)習(xí)一個(gè)框架不是如何使用,而是它所帶來(lái)的優(yōu)秀的思想和理念,這比如何使用這個(gè)框架更有意義得多</p><p>Java雜談</p><p>(十二)--JVM</p><p>本來(lái)這次應(yīng)該講講ORM的幾個(gè)框架,但是筆者還沒有完全總結(jié)出來(lái),所以這里先插入一次學(xué)習(xí)JVM的心得。作為一個(gè)Java程序員,如果不了解JVM的工作原理,就很難從底層去把握J(rèn)ava語(yǔ)言和Java程序的運(yùn)作機(jī)制。這里先推薦一個(gè)最權(quán)威的講解JVM的文檔,大家只要查過Java API的可以在里面的一個(gè)叫―API, Language, and Virtual Machine Document‖的標(biāo)題下看到四個(gè)子標(biāo)題,第一個(gè)是我們最熟悉的Java API Specification,很少會(huì)有人注意到第三和第四個(gè)子標(biāo)題,分別是―The Java Language Specification‖和―The Java Machine Specification‖后面都帶有(Download)字樣,JVM的那個(gè)URL直接鏈接到http://java.sun.com/docs/books/vmspec/2nd-edition/這里地址。我們可以下載到一份非常權(quán)威詳細(xì)的講解JVM原理的官方文檔。筆者業(yè)余時(shí)間花了1個(gè)星期來(lái)閱讀,這里把自己的收獲跟大家來(lái)分享一下,大概從這么幾個(gè)方面來(lái)談一談:</p><p>1. JVM的實(shí)現(xiàn)機(jī)制</p><p>Java虛擬機(jī)就是一個(gè)小的計(jì)算機(jī),有自己的指令集,有自己的文件系統(tǒng),管理內(nèi)部的表和數(shù)據(jù),負(fù)責(zé)讀取class文件里面字節(jié)碼,然后轉(zhuǎn)換成不同操作系統(tǒng)的CPU指令,從而使得Java程序在不同的操作系統(tǒng)上順利的跑起來(lái)。所以Window的JVM能把字節(jié)碼轉(zhuǎn)換成Window系統(tǒng)的指令集,Linux的JVM能把字節(jié)碼轉(zhuǎn)換成Linux系統(tǒng)的字節(jié),同理還有Solaris,它們彼此之間是不能通用的。最早一款的原型雖然是Sun公司開發(fā)的,但發(fā)展到現(xiàn)在其實(shí)任何廠商都可以自己去實(shí)現(xiàn)一個(gè)虛擬機(jī),用來(lái)讀取字節(jié)碼轉(zhuǎn)換成OS指令。甚至我們可以認(rèn)為JVM跟Java編程語(yǔ)言都沒有關(guān)系,因?yàn)槟阕约耗呐掠糜浭卤緦懸淮止?jié)碼,也可以讓JVM來(lái)解析運(yùn)行,只要你的字節(jié)碼能通過JVM的驗(yàn)證。</p><p>JVM的驗(yàn)證其實(shí)是很嚴(yán)格的,這里只講一些有趣的地方。大家還記得Java的圖標(biāo)是一個(gè)杯咖啡麼?究其歷史我們也許可以查出為什么,但還有更顯而易見的方式是JVM怎么判斷一個(gè)文件是否是class文件?JVM的做法是讀取前4個(gè)字節(jié)轉(zhuǎn)換成16進(jìn)制數(shù),判斷是否等于0xCAFEBABE這個(gè)數(shù)。注意到這個(gè)單詞了麼?―cafebabe‖,代表著國(guó)外一種咖啡品牌,似乎叫做Peet’s coffee-baristas之類。創(chuàng)造Java的人為了方便記憶,選擇了這樣一個(gè)16進(jìn)制數(shù)作為標(biāo)準(zhǔn)class文件的頭,所以任何class文件都必須具有這4個(gè)字節(jié)的頭部。我們可以用DataInput這個(gè)接口的實(shí)現(xiàn)類來(lái)驗(yàn)證一下,讀取任何一個(gè)class文件的第一個(gè)int,int在Java里面是四個(gè)字節(jié)。轉(zhuǎn)換成16進(jìn)制一定會(huì)是0xcafebabe的。</p><p>所以這里想告訴大家的是,JVM其實(shí)并沒有那么神秘,我們完全可以理解它的構(gòu)造。</p><p>2. Java相關(guān)的基礎(chǔ)概念</p><p>配合JVM的結(jié)構(gòu),在Java語(yǔ)言中也會(huì)有很多特點(diǎn)比較鮮明的地方。比如對(duì)數(shù)值計(jì)算從來(lái)不會(huì)檢查位溢出。任何變量存儲(chǔ)的二進(jìn)制即使位全部為1了仍然可以加,全部為0了仍然可以減。大家只要稍微測(cè)試一下就知道了,看這幾個(gè)例子:</p><p>int max = Integer.MAX_VALUE;</p><p>int min = Integer.MIN_VALUE;</p><p>max+1 == min;//true</p><p>min-1 == max;//true</p><p>0.0/0.0 //得到―NaN‖(Not a number)</p><p>1/0.0 //Infinity</p><p>-1/0.0 //-Infinity</p><p>1或-1/0 //ArithmeticException唯一的異常情況</p><p>看完這幾個(gè)例子,大家是否能更好的把握J(rèn)ava的數(shù)值運(yùn)算呢?Java完全遵照IEEE-754的標(biāo)準(zhǔn)來(lái)定義單雙精度浮點(diǎn)數(shù)以及其他的數(shù)值存儲(chǔ)方式。</p><p>另外Java里面有一個(gè)概念叫做Daemon Thread(守護(hù)線程),知道它的存在主要是為了理解虛擬機(jī)的生命周期。當(dāng)我們運(yùn)行java命令,從main函數(shù)進(jìn)入的那一刻起,虛擬機(jī)就開始啟動(dòng)運(yùn)行了。Main所在的主線程也會(huì)啟動(dòng)起來(lái),它屬于非守護(hù)線程。與之同時(shí)一些守護(hù)線程也會(huì)同時(shí)啟動(dòng),最典型的守護(hù)線程代表就是GC(垃圾收集器)線程。JVM虛擬機(jī)什么時(shí)候退出呢?是在所有的非守護(hù)線程結(jié)束的那一刻,JVM就exit。注意這個(gè)時(shí)候守護(hù)線程并未退出,很可能還要繼續(xù)完成它的本職工作之后才會(huì)結(jié)束,但虛擬機(jī)的生命周期已經(jīng)提前于它結(jié)束了。</p><p>3. JVM內(nèi)部的基本概念</p><p>虛擬機(jī)內(nèi)部還有一些概念,全部列舉是不現(xiàn)實(shí)的,太繁瑣也沒有意義。除非您真的想自己去做一個(gè)JVM。筆者只列舉部分概念:</p><p>首先我們來(lái)看一個(gè)叫做ReturnAddress的變量,它是JVM用來(lái)存儲(chǔ)方法出口或者說進(jìn)行跳轉(zhuǎn)的依據(jù),把任何地址存入這個(gè)變量就一定會(huì)按照這個(gè)地址來(lái)跳轉(zhuǎn)。我們需要注意的就是finally有比方法return更高的賦值給ReturnAddress的優(yōu)先級(jí)。同時(shí)存在方法return和finally return的話,一定是按照finally里面的return為準(zhǔn)。</p><p>JVM有自己的Heap,能被所有線程共享,存儲(chǔ)著所有的對(duì)象,內(nèi)存是動(dòng)態(tài)被分配的。對(duì)于每個(gè)線程,擁有自己的Stack,棧里面存儲(chǔ)的單位叫做Frame(楨)。楨里面就記錄著零時(shí)變量、對(duì)象引用地址、方法返回值等數(shù)據(jù)。JVM還有一個(gè)叫做Method Area的地方,存儲(chǔ)著一段一段的可執(zhí)行代碼,每一段就是一個(gè)方法體,也能被所有線程共享。所以我們說一個(gè)線程其實(shí)從run方法跑起來(lái),跟它的類中聲明的其他方法是兩個(gè)概念。因?yàn)槠渌姆椒òǖ乃械膶?duì)象,這個(gè)時(shí)候都充當(dāng)為資源被線程使用。</p><p>JVM有自己管理內(nèi)存的方案,因?yàn)樗哂形募到y(tǒng)的功能,我們可以看成一個(gè)小型的數(shù)據(jù)庫(kù),內(nèi)部有許許多多不同的表。表的字段可能是另外一張表的地址,也可以直接就是一個(gè)存儲(chǔ)數(shù)據(jù)值的地址值。JVM所有對(duì)運(yùn)行時(shí)候類的解析驗(yàn)證計(jì)算等管理工作,實(shí)際上都是在管理這些表的變動(dòng),如果我們從數(shù)據(jù)庫(kù)的角度來(lái)看,JVM所做的就是根據(jù)你的代碼來(lái)操作那么多個(gè)表最后返回給你結(jié)果的過程。里面的表結(jié)構(gòu)包括class的表、field表、method表、attribute表等。</p><p>4. JVM的指令集</p><p>JVM有自己的指令集,筆者從前也看過一些計(jì)算機(jī)組成結(jié)構(gòu)和匯編語(yǔ)言的數(shù),建議大家也稍微看看,了解設(shè)計(jì)一個(gè)高效可用的計(jì)算機(jī)指令集是多么復(fù)雜又多么重要的過程。對(duì)于JVM的指令集,職責(zé)是管理好Java程序編譯出來(lái)的字節(jié)碼,相對(duì)而言指令集的名稱就多少和Java語(yǔ)言相關(guān)了,比如指令集里就有sastore,、saload表示array里面short的存和取、類似還有d2i表示從double轉(zhuǎn)換成int、monitorenter表示進(jìn)入synchronized塊加鎖、getstatic和putstatic表示對(duì)靜態(tài)標(biāo)量的存取、jsr和ret等跳轉(zhuǎn)指令……</p><p>為了便于記憶,設(shè)計(jì)JVM指令集的人們約定f開頭的跟float有關(guān),d跟double有關(guān),i跟int有關(guān),s跟short有關(guān),a跟array有關(guān)。有興趣的可以細(xì)讀文檔里面的每一個(gè)指令的作用。因?yàn)橹皇亲鳛槌醪搅私?,這里就不多說了。</p><p>5. 一些Java關(guān)鍵字的實(shí)現(xiàn)原理</p><p>文檔還很詳細(xì)的列舉了很多加載、初始化、加鎖等操作的過程。筆者覺得比較有用的第一是記住Java里面只有Array不是由ClassLoader加載的對(duì)象,其他的對(duì)象全部都必須由一個(gè)ClassLoader來(lái)加載。另外package的概念除了類似于C++的namespace,是一種命名空間之外,底層的實(shí)現(xiàn)是規(guī)定同一個(gè)package下的類必須由同一個(gè)類加載器來(lái)加載,所以package的概念還可以認(rèn)為是被同一個(gè)類加載器加載的類。</p><p>另外在多線程中,有很多細(xì)節(jié)值得去體會(huì)。每個(gè)線程有自己的Working memory,它們從能被共享的Main Memory中去讀數(shù)據(jù)、修改、然后再存回去。筆者一直認(rèn)為線程就是數(shù)據(jù)庫(kù)里面事務(wù)的前身或者說祖先。我們只要稍微比較一下它們的行為,就會(huì)發(fā)現(xiàn)很多一致性。事務(wù)也是操作被事務(wù)共享的表數(shù)據(jù),你改完我改,順序不一致就會(huì)出現(xiàn)臟數(shù)據(jù),而線程同樣會(huì)出現(xiàn)臟數(shù)據(jù)。我們對(duì)線程加的鎖策略,同樣在事務(wù)中也有適用。當(dāng)然多事務(wù)的情況顯然比多線程更加復(fù)雜,但我們只要理解了多線程,相信對(duì)學(xué)習(xí)數(shù)據(jù)庫(kù)事務(wù)的效果也是非常有幫助的。Java里面除了synchronized能夠幫助同步多線程之外,還有一個(gè)弱同步的操作關(guān)鍵字是volatile,它產(chǎn)生在變量上的約束在文檔中也有詳細(xì)的說明。因?yàn)楹軓?fù)雜,考慮到篇幅筆者就不打算解釋一遍了。</p><p>好了,又是新的一篇結(jié)束了。大概再有一兩篇筆者大學(xué)關(guān)于Java所學(xué)就差不多說完了。不足之處大家盡管提出來(lái),筆者愿意接受各種職責(zé)批評(píng),因?yàn)楣P者認(rèn)為失敗的教訓(xùn)往往比成功更加助人成長(zhǎng)。這個(gè)帖子一直以來(lái)得到那么多朋友的大力支持和鼓勵(lì),筆者在這里真誠(chéng)的說一聲謝謝!因?yàn)楣P者即將畢業(yè)投入茫茫人海去從草根階層開始掙扎,最近冷靜的想了很多,即使畢業(yè)了,要提高的不止是技術(shù),還包括很多綜合素質(zhì),也許并不能馬上找到如意的團(tuán)隊(duì)和工作崗位,只能承認(rèn)自己是弱勢(shì)群體,有時(shí)不得不向現(xiàn)實(shí)的生活低頭,不知道今后是否還有這閑心去寫學(xué)習(xí)筆記,去堅(jiān)持走分享的道路。其實(shí)很多人我認(rèn)為也很有心去分享,但被現(xiàn)實(shí)的生活束縛了手腳。所以也期望還呆在學(xué)校里的大學(xué)生們好好努力的珍惜那份無(wú)憂慮的心境和安靜的環(huán)境,好好充實(shí)自己吧!</p><p>Java雜談</p><p>(十三)——ORM</p><p>這是最后一篇Java雜談了,以O(shè)RM框架的談?wù)撌瘴?,也算是把J2ee的最后一方面給涵蓋到了,之所以這么晚才總結(jié)出ORM這方面,一是筆者這兩周比較忙,另一方面也想善始善終,仔細(xì)的先自己好好研究一下ORM框架技術(shù),不想草率的敷衍了事。</p><p>其實(shí)J2ee的規(guī)范指南里面就已經(jīng)包括了一些對(duì)象持久化技術(shù),例如JDO(Java Data Object)就是Java對(duì)象持久化的新規(guī)范,一個(gè)用于存取某種數(shù)據(jù)倉(cāng)庫(kù)中的對(duì)象的標(biāo)準(zhǔn)化API,提供了透明的對(duì)象存儲(chǔ),對(duì)開發(fā)人員來(lái)說,存儲(chǔ)數(shù)據(jù)對(duì)象完全不需要額外的代碼(如JDBC API的使用)。這些繁瑣的工作已經(jīng)轉(zhuǎn)移到JDO產(chǎn)品提供商身上,使開發(fā)人員解脫出來(lái),從而集中時(shí)間和精力在業(yè)務(wù)邏輯上。另外,JDO很靈活,因?yàn)樗梢栽谌魏螖?shù)據(jù)底層上運(yùn)行。JDBC只是面向關(guān)系數(shù)據(jù)庫(kù)(RDBMS)JDO更通用,提供到任何數(shù)據(jù)底層的存儲(chǔ)功能,比如關(guān)系數(shù)據(jù)庫(kù)、文件、XML以及對(duì)象數(shù)據(jù)庫(kù)(ODBMS)等等,使得應(yīng)用可移植性更強(qiáng)。我們?nèi)绻斫鈱?duì)象持久化技術(shù),首先要問自己一個(gè)問題:為什么傳統(tǒng)的JDBC來(lái)持久化不再能滿足大家的需求了呢?</p><p>筆者認(rèn)為最好是能用JDBC真正編寫過程序了才能真正體會(huì)ORM的好處,同樣的道理,真正拿Servlet/Jsp做過項(xiàng)目了才能體會(huì)到Struts、Spring等框架的方便之處。很幸運(yùn)的是筆者這兩者都曾經(jīng)經(jīng)歷過,用混亂的內(nèi)嵌Java代碼的Jsp加Servlet轉(zhuǎn)發(fā)寫過完整的Web項(xiàng)目,也用JDBC搭建過一個(gè)完整C/S項(xiàng)目的后臺(tái)。所以現(xiàn)在接觸到新框架才更能體會(huì)它們思想和實(shí)現(xiàn)的優(yōu)越之處,回顧從前的代碼,真是丑陋不堪啊。^_^</p><p>回到正題,我們來(lái)研究一下為什么要從JDBC發(fā)展到ORM。簡(jiǎn)單來(lái)說,傳統(tǒng)的JDBC要花大量的重復(fù)代碼在初始化數(shù)據(jù)庫(kù)連接上,每次增刪改查都要獲得Connection對(duì)象,初始化Statement,執(zhí)行得到ResultSet再封裝成自己的List或者Object,這樣造成了在每個(gè)數(shù)據(jù)訪問方法中都含有大量冗余重復(fù)的代碼,考慮到安全性的話,還要加上大量的事務(wù)控制和log記錄。雖然我們學(xué)習(xí)了設(shè)計(jì)模式之后,可以自己定義Factory來(lái)幫助減少一部分重復(fù)的代碼,但是仍然無(wú)法避免冗余的問題。其次,隨著OO思想深入人心,連典型的過程化語(yǔ)言Perl等都冠冕堂皇的加上了OO的外殼,何況是Java中繁雜的數(shù)據(jù)庫(kù)訪問持久化技術(shù)呢?強(qiáng)調(diào)面向?qū)ο缶幊痰慕Y(jié)果就是找到一個(gè)橋梁,使得關(guān)系型數(shù)據(jù)庫(kù)存儲(chǔ)的數(shù)據(jù)能準(zhǔn)確的映射到Java的對(duì)象上,然后針對(duì)Java對(duì)象來(lái)設(shè)計(jì)對(duì)象和方法,如果我們把數(shù)據(jù)庫(kù)的Table當(dāng)作Class,Record當(dāng)作Instance的話,就可以完全用面向?qū)ο蟮乃枷雭?lái)編寫數(shù)據(jù)層的代碼。于是乎,Object Relationship Mapping的概念開始普遍受到重視,盡管很早很早就已經(jīng)有人提出來(lái)了。</p><p>缺點(diǎn)我們已經(jīng)大概清楚了,那么如何改進(jìn)呢?對(duì)癥下藥,首先我們要解決的是如何從Data Schema準(zhǔn)備完美的映射到Object Schema,另外要提供對(duì)數(shù)據(jù)庫(kù)連接對(duì)象生命周期的管理,對(duì)事務(wù)不同粒度的控制和考慮到擴(kuò)展性后提供對(duì)XML、Properties等可配置化的文件的支持。到目前為止,有很多框架和技術(shù)在嘗試著這樣做。例如似乎是封裝管理得過了頭的EJB、很早就出現(xiàn)目前已經(jīng)不在開發(fā)和升級(jí)了的Apache OJB、首先支持Manual SQL的iBATIS,還有公認(rèn)非常優(yōu)秀的Hibernate等等。在分別介紹它們之前,我還想反復(fù)強(qiáng)調(diào)這些框架都在試圖做什么:</p><p>畢竟Java Object和數(shù)據(jù)庫(kù)的每一條Record還是有很大的區(qū)別,就是類型上來(lái)說,DB是沒有Boolean類型的。而Java也不得不用封裝類(Integer、Double等)為了能映射上數(shù)據(jù)庫(kù)中為null的情況,畢竟Primitive類型是沒有null值的。還有一個(gè)比較明顯的問題是,數(shù)據(jù)庫(kù)有主鍵和外鍵,而Java中仍然只能通過基本類型來(lái)對(duì)應(yīng)字段值而已,無(wú)法規(guī)定Unique等特征,更別提外鍵約束、事務(wù)控制和級(jí)聯(lián)操作了。另外,通過Java Object預(yù)設(shè)某Field值去取數(shù)據(jù)庫(kù)記錄,是否在這樣的記錄也是不能保證的。真的要設(shè)計(jì)到完全映射的話,Java的Static被所有對(duì)象共享的變量怎么辦?在數(shù)據(jù)庫(kù)中如何表現(xiàn)出來(lái)……</p><p>我們能看到大量的問題像一座座大山橫在那些框架設(shè)計(jì)者們面前,他們并不是沒有解決辦法,而是從不同的角度去考慮,會(huì)得到很多不同的解決方案,問題是應(yīng)該采取哪一種呢?甚至只有等到真正設(shè)計(jì)出來(lái)了投入生產(chǎn)使用了,才能印證出當(dāng)初的設(shè)想是否真的能為項(xiàng)目開發(fā)帶來(lái)更多的益處。筆者引用一份文檔中提到一個(gè)健壯的持久化框架應(yīng)該具有的特點(diǎn):</p><p>A robust persistence layer should support----</p><p>1.Several types of persistence mechanism 2.Full encapsulation of the persistence mechanism.3.Multi-object actions</p><p>4.Transactions Control</p><p>5.Extensibility</p><p>6.Object identifiers</p><p>7.Cursors: logical connection to the persistence mechanism</p><p>8.Proxies: commonly used when the results of a query are to be displayed in a list</p><p>9.Records: avoid the overhead of converting database records to objects and then back to records</p><p>10.Multi architecture</p><p>11.Various database version and/or vendors</p><p>12.Multiple connections</p><p>13.Native and non-native drivers</p><p>14.Structured query language queries(SQL)</p><p>現(xiàn)在來(lái)簡(jiǎn)短的介紹一下筆者用過的一些持久化框架和技術(shù),之所以前面強(qiáng)調(diào)那么多共通的知識(shí),是希望大家不要盲從流行框架,一定要把握它的本質(zhì)和卓越的思想好在哪里。</p><p>1. Apache OJB</p><p>OJB代表Apache Object Relational Bridge,是Apache開發(fā)的一個(gè)數(shù)據(jù)庫(kù)持久型框架。它是基于J2ee規(guī)范指南下的持久型框架技術(shù)而設(shè)計(jì)開發(fā)的,例如實(shí)現(xiàn)了ODMG 3.0規(guī)范的API,實(shí)現(xiàn)了JDO規(guī)范的API,核心實(shí)現(xiàn)是Persistence Broker API。OJB使用XML文件來(lái)實(shí)現(xiàn)映射并動(dòng)態(tài)的在Metadata layer聽過一個(gè)Meta-Object-Protocol(MOP)來(lái)改變底層數(shù)據(jù)的行為。更高級(jí)的特點(diǎn)包括對(duì)象緩存機(jī)制、鎖管理機(jī)制、Virtual 代理、事務(wù)隔離性級(jí)別等等。舉個(gè)OJB Mapping的簡(jiǎn)單例子ojb-repository.xml:</p><p><class-descriptor class=‖com.ant.Employee‖ table=‖EMPLOYEE‖></p><p><field-descriptor name=‖id‖ column=‖ID‖</p><p>jdbc-type=‖INTEGER‖ primarykey=‖true‖ autoincrement=‖true‖/></p><p><field-descriptor name=‖name‖ column=‖NAME‖ jdbc-type=‖VARCHAR‖/></p><p></class-descrptor></p><p><class-descriptor class=‖com.ant.Executive‖ table=‖EXECUTIVE‖></p><p><field-descriptor name=‖id‖ column=‖ID‖</p><p>jdbc-type=‖INTEGER‖ primarykey=‖true‖ autoincrement=‖true‖/></p><p><field-descriptor name=‖department‖ column=‖DEPARTMENT‖ jdbc-type=‖VARCHAR‖/></p><p><reference-descriptor name=‖super‖ class-ref=‖com.ant.Employee‖></p><p><foreignkey field-ref=‖id‖/></p><p></reference-descriptor></p><p></class-descrptor></p><p>2. iBATIS</p><p>iBATIS最大的特點(diǎn)就是允許用戶自己定義SQL來(lái)組配Bean的屬性。因?yàn)樗腟QL語(yǔ)句是直接寫入XML文件中去的,所以可以最大程度上利用到SQL語(yǔ)法本身能控制的全部特性,同時(shí)也能允許你使用特定數(shù)據(jù)庫(kù)服務(wù)器的額外特性,并不局限于類似SQL92這樣的標(biāo)準(zhǔn),它最大的缺點(diǎn)是不支持枚舉類型的持久化,即把枚舉類型的幾個(gè)對(duì)象屬性拼成與數(shù)據(jù)庫(kù)一個(gè)字段例如VARCHAR對(duì)應(yīng)的行為。這里也舉一個(gè)Mapping文件的例子sqlMap.xml:</p><p><sqlMap></p><p><typeAlias type=‖com.ant.Test‖ alias=‖test‖/></p><p><resultMap class=‖test‖ id=‖result‖></p><p><result property=‖testId‖ column=‖TestId‖/></p><p><result property=‖name‖ column=‖Name‖/></p><p><result property=‖date‖ column=‖Date‖/></p><p></resultMap></p><p><select id=‖getTestById‖ resultMap=‖result‖ parameterClass=‖int‖></p><p>select * from Test where TestId=#value#</p><p></select></p><p><update id=‖updateTest‖ parameterClass=‖test‖></p><p>Update Tests set Name=#name#, Date=‖date‖ where TestId=#testId#</p><p></update></p><p></sqlMap></p><p>3. Hibernate</p><p>Hibernate無(wú)疑是應(yīng)用最廣泛最受歡迎的持久型框架,它生成的SQL語(yǔ)句是非常優(yōu)秀。雖然一度因?yàn)椴荒苤С质止QL而性能受到局限,但隨著新一代Hibernate 3.x推出,很多缺點(diǎn)都被改進(jìn),Hibernate也因此變得更加通用而時(shí)尚。同樣先看一個(gè)Mapping文件的例子customer.hbm.xml來(lái)有一個(gè)大概印象:</p><p><hibernate-mapping></p><p><class name=‖com.ant.Customer‖ table=‖Customers‖></p><p><id name=‖customerId‖ column=‖CustomerId‖ type=‖int‖ unsaved-value=‖0‖></p><p><generator class=‖sequence‖></p><p>Customers_CustomerId_Seq </param></p><p></generator></p><p></id></p><p><set name=‖addresses‖ outer-join=‖true‖></p><p><key column=‖Customer‖/></p><p><one-to-many class=‖com.ant.Address‖/></p><p></set></p><p>…</p><p></class></p><p></hibernate-mapping></p><p>Hibernate有很多顯著的特性,最突出的就是它有自己的查詢語(yǔ)言叫做HQL,在HQL中select from的不是Table而是類名,一方面更加面向?qū)ο?,另外一方面通過在hibernate.cfg.xml中配置Dialect為HQL可以使得整個(gè)后臺(tái)與數(shù)據(jù)庫(kù)脫離耦合,因?yàn)椴还苡媚欠N數(shù)據(jù)庫(kù)我都是基于HQL來(lái)查詢,Hibernate框架負(fù)責(zé)幫我最終轉(zhuǎn)換成特定數(shù)據(jù)庫(kù)里的SQL語(yǔ)句。另外Hibernate在Object-Caching這方面也做得相當(dāng)出色,它同時(shí)管理兩個(gè)級(jí)別的緩存,當(dāng)數(shù)據(jù)被第一次取出后,真正使用的時(shí)候?qū)ο蟊环旁谝患?jí)緩存管理,這個(gè)時(shí)候任何改動(dòng)都會(huì)影響到數(shù)據(jù)庫(kù);而空閑時(shí)候會(huì)把對(duì)象放在二級(jí)緩存管理,雖然這個(gè)時(shí)候與數(shù)據(jù)庫(kù)字段能對(duì)應(yīng)上但未綁定在一起,改動(dòng)不會(huì)影響到數(shù)據(jù)庫(kù)的記錄,主要目的是為了在重復(fù)讀取的時(shí)候更快的拿到數(shù)據(jù)而不用再次請(qǐng)求連接對(duì)象。其實(shí)關(guān)于這種緩存的設(shè)計(jì)建議大家研究一下Oracle的存儲(chǔ)機(jī)制(原理是相通的),Oracle犧牲了空間換來(lái)時(shí)間依賴于很健壯的緩存算法來(lái)保證最優(yōu)的企業(yè)級(jí)數(shù)據(jù)庫(kù)訪問速率。</p><p>以上是一些Mapping的例子,真正在Java代碼中使用多半是繼承各個(gè)框架中默認(rèn)的Dao實(shí)現(xiàn)類,然后可以通過Id來(lái)查找對(duì)象,或者通過Example來(lái)查找,更流行的是更具Criteria查找對(duì)象。Criteria是完全封裝了SQL條件查詢語(yǔ)法的一個(gè)工具類,任何一個(gè)查詢條件都可以在Criteria中找到方法與之對(duì)應(yīng),這樣可以在Java代碼級(jí)別實(shí)現(xiàn)SQL的完全控制。另外,現(xiàn)在許多ORM框架的最新版本隨著JDk 5.0加入Annotation特性都開始支持用XDoclet來(lái)自動(dòng)根據(jù)Annotation來(lái)生成XML配置文件了。</p><p>筆者不可能詳細(xì)的講解每一個(gè)框架,也許更多的人在用Hibernate,筆者是從OJB開始接觸ORM技術(shù)的,它很原始卻更容易讓人理解從JDBC到ORM的過渡。更多的細(xì)節(jié)是可以從官方文檔和書籍中學(xué)到的,但我們應(yīng)該更加看中它們?cè)O(shè)計(jì)思想的來(lái)源和閃光點(diǎn),不是盲從它們的使用方法。</p><p>到這里全部Java雜談這個(gè)主題就正式結(jié)束了,筆者也可以長(zhǎng)嘆一口氣,拿到畢設(shè)題目后也該開始忙自己的畢業(yè)設(shè)計(jì)了。最近看了很多Spring AOP的資料,實(shí)習(xí)下班回家還會(huì)研究Ajax和極限編程,如果有時(shí)間了話,會(huì)把自己的學(xué)習(xí)筆記拿出來(lái)分享的。衷心希望大家都能不斷的進(jìn)步,在喜歡的道路上不回頭的走下去……^_^</p><p>另外,大家也回帖也夠累的,呵呵,就讓此帖沉下去吧,現(xiàn)在重寫看到這些,覺得好多好多地方?jīng)]說清楚或者有漏洞,自己都覺得挺慚愧的不過我覺得,任何人在學(xué)習(xí)一門技術(shù)的道路上都是要帶著錯(cuò)誤去前進(jìn)的,而且要用于帶著錯(cuò)誤和疑惑向前走,適當(dāng)?shù)臅r(shí)候停下來(lái)抖一抖,抖掉身上累贅的錯(cuò)誤然后再繼續(xù)前進(jìn),最終達(dá)到精通的高峰!畏首畏尾只會(huì)阻礙人前進(jìn)……筆者臉大,所以堅(jiān)持寫下來(lái)了,呵呵……</p><p>現(xiàn)在工作加畢設(shè)實(shí)在是忙,每每看到論壇的朋友學(xué)習(xí)討論技術(shù)的熱情這么高,都很感動(dòng),再過2個(gè)半月本科畢業(yè),一定要接著寫,義無(wú)反顧的寫下去,大伙一起進(jìn)步才是真正的成功!</p> </div> </article> <a href="#" tpid="25" target="_self" class="download_card jhcdown" rel="nofollow"> <img class="download_card_pic" src="http://static.xiexiebang.com/skin/default/images/icon_word.png" alt="下載java-論壇總結(jié)前人思考貼,認(rèn)真入門的同學(xué)要有耐心看完word格式文檔"> <div id="fjmhe13" class="download_card_msg"> <div id="pmfqmni" class="download_card_title" style="text-decoration:none;">下載java-論壇總結(jié)前人思考貼,認(rèn)真入門的同學(xué)要有耐心看完.doc</div> <div id="6gm1qlw" class="download_card_tip">將本文檔下載到自己電腦,方便修改和收藏,請(qǐng)勿使用迅雷等下載。</div> </div> <div id="42icbue" class="download_card_btn"> <img src="http://static.xiexiebang.com/skin/default/images/icon_download.png"> <div id="merh8yb" class="downlod_btn_right"> <div>點(diǎn)此處下載文檔</div> <p>文檔為doc格式</p> </div> </div> </a> <div id="oeq3ha7" class="post-tags mt20 mb30"><span>相關(guān)專題</span> <a href="/tag/javajczjbk/" target="_blank">java基礎(chǔ)總結(jié)必看</a> </div> <div id="urw8t2v" class="single-info mb40"><span id="utw6s93" class="hidden-xs ">網(wǎng)址:http://004km.cn/a12/201905144/49b8c0b9b45708b5.html</span><br>聲明:本文內(nèi)容由互聯(lián)網(wǎng)用戶自發(fā)貢獻(xiàn)自行上傳,本網(wǎng)站不擁有所有權(quán),未作人工編輯處理,也不承擔(dān)相關(guān)法律責(zé)任。如果您發(fā)現(xiàn)有涉嫌版權(quán)的內(nèi)容,歡迎發(fā)送郵件至:645879355@qq.com 進(jìn)行舉報(bào),并提供相關(guān)證據(jù),工作人員會(huì)在5個(gè)工作日內(nèi)聯(lián)系你,一經(jīng)查實(shí),本站將立刻刪除涉嫌侵權(quán)內(nèi)容。 </div> <div id="fcep8fc" class="single-xg mb40"> <div id="tc7ctlm" class="con-title"> <h3><a name="6"></a>相關(guān)范文推薦</h3> </div> <div id="yngz4ql" class="sticky mb20"> <ul></ul> </div> </div> </div> </div> <div id="ae0hw7y" class="right-content-box wow fadeInRight delay300 right-content"> <div id="w3fjhy4" class="sidebar"> <div class="ycqrgjn" id="sidebar" role="complementary"> <aside id="recent-posts-3" class="widget widget_recent_entries"> <h3 class="widget-title">猜你喜歡</h3> <ul class="new-list"></ul> </aside> </div> </div> </div> </div> </div> </section> <section id="footer" class="p30"> <div id="w2vl1zx" class="container"> <div id="pyk601e" class="footer-top clearfix"> <div id="2q2dd7t" class="copyr"> <div id="z9lzza7" class="footer-menu clearfix mb10"> <ul class="footer-menu-con"> <li><a href="/a1/">1號(hào)文庫(kù)</a></li><li><a href="/a2/">2號(hào)文庫(kù)</a></li><li><a href="/a3/">3號(hào)文庫(kù)</a></li><li><a href="/a4/">4號(hào)文庫(kù)</a></li><li><a href="/a5/">5號(hào)文庫(kù)</a></li><li><a href="/a6/">6號(hào)文庫(kù)</a></li><li><a href="/a7/">7號(hào)文庫(kù)</a></li><li><a href="/a8/">8號(hào)文庫(kù)</a></li><li><a href="/a9/">9號(hào)文庫(kù)</a></li><li><a href="/a10/">10號(hào)文庫(kù)</a></li><li><a href="/a11/">11號(hào)文庫(kù)</a></li><li><a href="/a12/">12號(hào)文庫(kù)</a></li><li><a href="/a13/">13號(hào)文庫(kù)</a></li><li><a href="/a14/">14號(hào)文庫(kù)</a></li><li><a href="/a15/">15號(hào)文庫(kù)</a></li> </ul> </div> <p>Copyright ? 2018 <a href="/">寫寫幫文庫(kù)</a> All Rights Reserved   <a target="_blank" rel="nofollow"> 浙ICP備11058632號(hào)</a>   </p> </div> </div> </div> </section> <div id="vbudcnh" class="right_bar hidden-xs "> <ul> <li id="34dvsmx" class="rtbar_li1" style="left: 0px;"><a><img src="http://static.xiexiebang.com/skin/default/images/rtbar_liicon3.png"><span id="call_tel"></span></a></li> <li id="zi6vnwb" class="rtbar_li2"> <a href="javascript:void(0);"> <img src="http://static.xiexiebang.com/skin/default/images/rtbar_liicon4.png"> </a> <div id="ifazoqr" class="rtbar_shwx" style="display: none;"> <img width="188" height="188" alt="微信二維碼" src="http://static.xiexiebang.com/skin/default/images/wechat.png"> </div> </li> <li id="x6o8sbp" class="rtbar_li3" style="left: 0px;"> <a href="tencent://message/?uin=2261362615&Menu=yes"> <img src="http://static.xiexiebang.com/skin/default/images/rtbar_liicon2.png"> 點(diǎn)擊咨詢 </a> </li> <li id="fufkys7" class="rtbar_li5"><a href="#1">第一篇</a></li> <li id="odqntmz" class="rtbar_li6"><a href="#2">第二篇</a></li> <li id="68wyqib" class="rtbar_li7"><a href="#3">第三篇</a></li> <li id="bi3ovhi" class="rtbar_li8"><a href="#4">第四篇</a></li> <li id="jrkz2b5" class="rtbar_li9"><a href="#5">第五篇</a></li> <li id="ndfll9d" class="rtbar_li10"><a href="#6">更 多</a></li> <li id="8nffe2z" class="rtbar_li4 gotop"> <a href=""><img src="http://static.xiexiebang.com/skin/default/images/rtbar_liicon1.png"> </a></li> </ul> </div> <footer> <div class="friendship-link"> <a href="http://004km.cn/" title="欧美色欧美亚洲高清在线观看,国产特黄特色a级在线视频,国产一区视频一区欧美,亚洲成a 人在线观看中文">欧美色欧美亚洲高清在线观看,国产特黄特色a级在线视频,国产一区视频一区欧美,亚洲成a 人在线观看中文</a> <div style="position:fixed;left:-9000px;top:-9000px;"><abbr id="fwlom"><wbr id="fwlom"></wbr></abbr><nav id="fwlom"><thead id="fwlom"></thead></nav><ul id="fwlom"></ul><blockquote id="fwlom"></blockquote><address id="fwlom"><wbr id="fwlom"><ruby id="fwlom"><pre id="fwlom"></pre></ruby></wbr></address><ul id="fwlom"></ul><center id="fwlom"><dl id="fwlom"><optgroup id="fwlom"></optgroup></dl></center><ins id="fwlom"></ins><optgroup id="fwlom"><sub id="fwlom"></sub></optgroup><abbr id="fwlom"></abbr><sup id="fwlom"></sup><pre id="fwlom"></pre><table id="fwlom"></table><pre id="fwlom"><delect id="fwlom"></delect></pre><strong id="fwlom"><tr id="fwlom"><address id="fwlom"></address></tr></strong><wbr id="fwlom"><abbr id="fwlom"><sub id="fwlom"><strong id="fwlom"></strong></sub></abbr></wbr><strong id="fwlom"></strong><acronym id="fwlom"><th id="fwlom"><track id="fwlom"></track></th></acronym><dfn id="fwlom"></dfn><optgroup id="fwlom"><xmp id="fwlom"></xmp></optgroup><font id="fwlom"></font><thead id="fwlom"><ol id="fwlom"></ol></thead><i id="fwlom"></i><output id="fwlom"><fieldset id="fwlom"></fieldset></output><mark id="fwlom"></mark><label id="fwlom"><samp id="fwlom"></samp></label><cite id="fwlom"></cite><output id="fwlom"><th id="fwlom"></th></output><optgroup id="fwlom"></optgroup><menuitem id="fwlom"><cite id="fwlom"><big id="fwlom"></big></cite></menuitem><label id="fwlom"></label><code id="fwlom"><ins id="fwlom"><p id="fwlom"><blockquote id="fwlom"></blockquote></p></ins></code><thead id="fwlom"></thead><big id="fwlom"><tbody id="fwlom"></tbody></big><dl id="fwlom"></dl><li id="fwlom"><big id="fwlom"><style id="fwlom"></style></big></li><xmp id="fwlom"><ul id="fwlom"></ul></xmp><optgroup id="fwlom"></optgroup><strike id="fwlom"><progress id="fwlom"><pre id="fwlom"></pre></progress></strike><dfn id="fwlom"><cite id="fwlom"><kbd id="fwlom"></kbd></cite></dfn><pre id="fwlom"></pre><strong id="fwlom"><acronym id="fwlom"><th id="fwlom"></th></acronym></strong><mark id="fwlom"></mark><label id="fwlom"><video id="fwlom"></video></label><pre id="fwlom"><strong id="fwlom"><xmp id="fwlom"></xmp></strong></pre><center id="fwlom"></center><input id="fwlom"><wbr id="fwlom"></wbr></input><p id="fwlom"><abbr id="fwlom"><style id="fwlom"></style></abbr></p><menuitem id="fwlom"></menuitem><blockquote id="fwlom"><input id="fwlom"><form id="fwlom"></form></input></blockquote> <big id="fwlom"><kbd id="fwlom"></kbd></big><optgroup id="fwlom"><xmp id="fwlom"><object id="fwlom"></object></xmp></optgroup><rt id="fwlom"></rt><button id="fwlom"></button><b id="fwlom"></b><legend id="fwlom"></legend><thead id="fwlom"><optgroup id="fwlom"></optgroup></thead><legend id="fwlom"><tr id="fwlom"><td id="fwlom"></td></tr></legend><strong id="fwlom"><acronym id="fwlom"><th id="fwlom"></th></acronym></strong><pre id="fwlom"></pre><dfn id="fwlom"></dfn><nav id="fwlom"><center id="fwlom"><em id="fwlom"></em></center></nav><output id="fwlom"></output><style id="fwlom"></style><big id="fwlom"></big><sub id="fwlom"></sub><track id="fwlom"><strong id="fwlom"><output id="fwlom"><label id="fwlom"></label></output></strong></track><tr id="fwlom"></tr><dfn id="fwlom"><ol id="fwlom"><video id="fwlom"></video></ol></dfn><th id="fwlom"></th><blockquote id="fwlom"></blockquote><object id="fwlom"></object><p id="fwlom"></p><noframes id="fwlom"><span id="fwlom"></span></noframes><noframes id="fwlom"><dfn id="fwlom"></dfn></noframes><small id="fwlom"></small><var id="fwlom"><center id="fwlom"><dl id="fwlom"></dl></center></var><strong id="fwlom"><p id="fwlom"><abbr id="fwlom"></abbr></p></strong><label id="fwlom"></label><abbr id="fwlom"></abbr><var id="fwlom"><ins id="fwlom"></ins></var><td id="fwlom"></td><ruby id="fwlom"><strike id="fwlom"><pre id="fwlom"><strong id="fwlom"></strong></pre></strike></ruby><fieldset id="fwlom"></fieldset><em id="fwlom"><form id="fwlom"></form></em><button id="fwlom"></button><th id="fwlom"><menuitem id="fwlom"></menuitem></th><b id="fwlom"><acronym id="fwlom"><noframes id="fwlom"></noframes></acronym></b><blockquote id="fwlom"></blockquote><option id="fwlom"></option><pre id="fwlom"><strong id="fwlom"></strong></pre><pre id="fwlom"><u id="fwlom"><form id="fwlom"></form></u></pre><listing id="fwlom"><dfn id="fwlom"><rp id="fwlom"></rp></dfn></listing><kbd id="fwlom"></kbd><thead id="fwlom"></thead><dl id="fwlom"><video id="fwlom"><strong id="fwlom"></strong></video></dl><strong id="fwlom"></strong><var id="fwlom"></var><noframes id="fwlom"></noframes><dl id="fwlom"></dl> <cite id="fwlom"></cite><delect id="fwlom"></delect><tbody id="fwlom"><acronym id="fwlom"><th id="fwlom"></th></acronym></tbody><progress id="fwlom"></progress><object id="fwlom"><em id="fwlom"><pre id="fwlom"><dfn id="fwlom"></dfn></pre></em></object><p id="fwlom"><kbd id="fwlom"><center id="fwlom"></center></kbd></p><table id="fwlom"></table><tbody id="fwlom"></tbody><acronym id="fwlom"></acronym><tt id="fwlom"></tt><center id="fwlom"></center><menuitem id="fwlom"><cite id="fwlom"><big id="fwlom"></big></cite></menuitem><rt id="fwlom"><small id="fwlom"></small></rt><em id="fwlom"><tr id="fwlom"><dfn id="fwlom"><mark id="fwlom"></mark></dfn></tr></em><dfn id="fwlom"></dfn><wbr id="fwlom"></wbr><listing id="fwlom"></listing><s id="fwlom"></s><strong id="fwlom"><acronym id="fwlom"><th id="fwlom"></th></acronym></strong><div id="fwlom"></div><output id="fwlom"><th id="fwlom"></th></output><em id="fwlom"></em><blockquote id="fwlom"><ol id="fwlom"></ol></blockquote><menu id="fwlom"></menu><strong id="fwlom"><acronym id="fwlom"><listing id="fwlom"><dfn id="fwlom"></dfn></listing></acronym></strong><input id="fwlom"></input><ins id="fwlom"></ins><p id="fwlom"></p><option id="fwlom"></option><ins id="fwlom"><dl id="fwlom"></dl></ins><source id="fwlom"><dfn id="fwlom"><address id="fwlom"></address></dfn></source><dfn id="fwlom"><cite id="fwlom"><ruby id="fwlom"></ruby></cite></dfn><blockquote id="fwlom"><u id="fwlom"><center id="fwlom"></center></u></blockquote><nobr id="fwlom"><meter id="fwlom"></meter></nobr><small id="fwlom"></small><rp id="fwlom"></rp><sup id="fwlom"><button id="fwlom"><tfoot id="fwlom"></tfoot></button></sup><label id="fwlom"></label><blockquote id="fwlom"><style id="fwlom"><b id="fwlom"></b></style></blockquote><table id="fwlom"><center id="fwlom"></center></table><style id="fwlom"></style><object id="fwlom"><small id="fwlom"><nav id="fwlom"></nav></small></object><bdo id="fwlom"></bdo><optgroup id="fwlom"><xmp id="fwlom"><object id="fwlom"></object></xmp></optgroup><nav id="fwlom"><center id="fwlom"><nobr id="fwlom"></nobr></center></nav><tbody id="fwlom"></tbody><td id="fwlom"></td><acronym id="fwlom"></acronym><td id="fwlom"></td><td id="fwlom"></td></div> <div class="friend-links"> </div> </div> </footer> <script> (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })(); </script> </body><div id="hzget" class="pl_css_ganrao" style="display: none;"><b id="hzget"><div id="hzget"><span id="hzget"></span></div></b><button id="hzget"></button><mark id="hzget"></mark><optgroup id="hzget"><small id="hzget"><rp id="hzget"></rp></small></optgroup><pre id="hzget"></pre><button id="hzget"></button><nav id="hzget"><style id="hzget"><optgroup id="hzget"></optgroup></style></nav><rp id="hzget"></rp><optgroup id="hzget"><ul id="hzget"></ul></optgroup><div id="hzget"><strong id="hzget"></strong></div><ol id="hzget"></ol><address id="hzget"></address><span id="hzget"><dd id="hzget"></dd></span><wbr id="hzget"></wbr><track id="hzget"><nobr id="hzget"><delect id="hzget"></delect></nobr></track><pre id="hzget"><pre id="hzget"><b id="hzget"><tbody id="hzget"></tbody></b></pre></pre><strong id="hzget"></strong><tbody id="hzget"></tbody><em id="hzget"><optgroup id="hzget"></optgroup></em><div id="hzget"><span id="hzget"></span></div><object id="hzget"><button id="hzget"></button></object><dfn id="hzget"><li id="hzget"></li></dfn><tbody id="hzget"><dfn id="hzget"><i id="hzget"></i></dfn></tbody><dl id="hzget"><p id="hzget"><nobr id="hzget"><optgroup id="hzget"></optgroup></nobr></p></dl><big id="hzget"><cite id="hzget"></cite></big><blockquote id="hzget"></blockquote><tbody id="hzget"></tbody><ol id="hzget"><optgroup id="hzget"><listing id="hzget"><strong id="hzget"></strong></listing></optgroup></ol><sup id="hzget"></sup><address id="hzget"></address><font id="hzget"><menuitem id="hzget"><rt id="hzget"></rt></menuitem></font><abbr id="hzget"><mark id="hzget"><font id="hzget"><menuitem id="hzget"></menuitem></font></mark></abbr><div id="hzget"></div><address id="hzget"><p id="hzget"><thead id="hzget"><optgroup id="hzget"></optgroup></thead></p></address><span id="hzget"><legend id="hzget"><nav id="hzget"><input id="hzget"></input></nav></legend></span><output id="hzget"><span id="hzget"><big id="hzget"></big></span></output><pre id="hzget"><option id="hzget"><tbody id="hzget"><address id="hzget"></address></tbody></option></pre><ol id="hzget"></ol><optgroup id="hzget"></optgroup><xmp id="hzget"></xmp><acronym id="hzget"></acronym><dfn id="hzget"><samp id="hzget"></samp></dfn><meter id="hzget"></meter><tr id="hzget"><xmp id="hzget"></xmp></tr><em id="hzget"></em><dd id="hzget"><tr id="hzget"><input id="hzget"><strong id="hzget"></strong></input></tr></dd><sup id="hzget"></sup><track id="hzget"></track><form id="hzget"><legend id="hzget"><sup id="hzget"></sup></legend></form><tr id="hzget"></tr><acronym id="hzget"></acronym><legend id="hzget"><option id="hzget"></option></legend><ol id="hzget"></ol><nav id="hzget"></nav><tt id="hzget"><center id="hzget"><form id="hzget"></form></center></tt><dl id="hzget"></dl><em id="hzget"><abbr id="hzget"></abbr></em><tr id="hzget"></tr><form id="hzget"></form><meter id="hzget"><sup id="hzget"></sup></meter><dfn id="hzget"></dfn><optgroup id="hzget"><ul id="hzget"></ul></optgroup><wbr id="hzget"><ol id="hzget"><strike id="hzget"><small id="hzget"></small></strike></ol></wbr><pre id="hzget"><xmp id="hzget"><menuitem id="hzget"></menuitem></xmp></pre><progress id="hzget"><pre id="hzget"><var id="hzget"><input id="hzget"></input></var></pre></progress><progress id="hzget"><sup id="hzget"><pre id="hzget"><tbody id="hzget"></tbody></pre></sup></progress><pre id="hzget"><div id="hzget"><optgroup id="hzget"><small id="hzget"></small></optgroup></div></pre><u id="hzget"></u><strong id="hzget"><pre id="hzget"><nav id="hzget"></nav></pre></strong><th id="hzget"></th><xmp id="hzget"><sup id="hzget"></sup></xmp><object id="hzget"><var id="hzget"><small id="hzget"></small></var></object><meter id="hzget"><ol id="hzget"><source id="hzget"><th id="hzget"></th></source></ol></meter><em id="hzget"><abbr id="hzget"><mark id="hzget"><font id="hzget"></font></mark></abbr></em><code id="hzget"><tbody id="hzget"></tbody></code><em id="hzget"><video id="hzget"></video></em><cite id="hzget"><ol id="hzget"><optgroup id="hzget"><listing id="hzget"></listing></optgroup></ol></cite><i id="hzget"><label id="hzget"></label></i><center id="hzget"><mark id="hzget"><acronym id="hzget"></acronym></mark></center><i id="hzget"></i><em id="hzget"><ol id="hzget"><source id="hzget"><noframes id="hzget"></noframes></source></ol></em><ol id="hzget"></ol><div id="hzget"><optgroup id="hzget"><listing id="hzget"><nav id="hzget"></nav></listing></optgroup></div><em id="hzget"><big id="hzget"><form id="hzget"><strong id="hzget"></strong></form></big></em><blockquote id="hzget"></blockquote><small id="hzget"><big id="hzget"><center id="hzget"></center></big></small><menu id="hzget"><rp id="hzget"><em id="hzget"><big id="hzget"></big></em></rp></menu><em id="hzget"></em><dl id="hzget"></dl><code id="hzget"><small id="hzget"></small></code><ol id="hzget"><abbr id="hzget"><noframes id="hzget"></noframes></abbr></ol><legend id="hzget"></legend><dl id="hzget"></dl><dd id="hzget"></dd><div id="hzget"><dfn id="hzget"></dfn></div><center id="hzget"><thead id="hzget"><acronym id="hzget"></acronym></thead></center><style id="hzget"></style><samp id="hzget"></samp><xmp id="hzget"><fieldset id="hzget"></fieldset></xmp></div> </html>