給學(xué)Java的大學(xué)生們分享一些經(jīng)驗(yàn).doc_第1頁
給學(xué)Java的大學(xué)生們分享一些經(jīng)驗(yàn).doc_第2頁
給學(xué)Java的大學(xué)生們分享一些經(jīng)驗(yàn).doc_第3頁
給學(xué)Java的大學(xué)生們分享一些經(jīng)驗(yàn).doc_第4頁
給學(xué)Java的大學(xué)生們分享一些經(jīng)驗(yàn).doc_第5頁
已閱讀5頁,還剩34頁未讀, 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)

文檔簡(jiǎn)介

想來學(xué)習(xí)Java也有兩個(gè)年頭了,永遠(yuǎn)不敢說多么精通,但也想談?wù)勛约旱母惺?,寫給軟件學(xué)院的同仁們,幫助大家在技術(shù)的道路上少一點(diǎn)彎路。說得偉大一點(diǎn)是希望大家為軟件學(xué)院爭(zhēng)氣,其實(shí)最主要的還是大家自身的進(jìn)步提升 1 關(guān)于動(dòng)態(tài)加載機(jī)制 學(xué)習(xí)Java比C+更容易理解OOP的思想,畢竟C+還混合了不少面向過程的成分。很多人都能背出來Java語言的特點(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)來的順序。即使你聲明了一個(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() 打印出來就可以看到,把class文件放在不同的路徑下再次執(zhí)行,就會(huì)看到區(qū)別。特別注意的是如果打印出來是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類庫的class文件,都在lib目錄下打包成了jar。大家可以自己驗(yàn)證。至于在windows上的虛擬機(jī)是哪個(gè)文件呢? 學(xué)過MFC的都知道什么是dll文件吧,那么大家看看jre/bin/client里面是不是有一個(gè)jvm.dll呢?那就是虛擬機(jī)。 Jdk 是java development kit,是java的開發(fā)工具包,里面包含了各種類庫和工具。當(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ā)無關(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)境變量中啊? 試想一下如果java為了提供給大多數(shù)人使用,他們是不需要jdk做開發(fā)的,只需要jre能讓java程序跑起來就可以了,那么每個(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ù)寫下去,好不容易等到國慶黃金周,實(shí)習(xí)總算有一個(gè)休息的階段,于是這就開始寫第二篇了。希望這次寫的仍然對(duì)志同道合的朋友們有所幫助。上回講了Java動(dòng)態(tài)加載機(jī)制、classLoader原理和關(guān)于jdk和jre三個(gè)問題。這次延續(xù)著講一些具體的類庫 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)該先理解清楚它的來源接口,不管在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ū)別?新集合框架大家可以在這些包中找sincejdk1.2的,之前的如vector和Hashtable都是舊的集合框架包括的類。那么區(qū)別是? a.新集合框架的命名更加科學(xué)合理。例如List下的ArrayList和LinkedList b.新集合框架下全部都是非線程安全的。建議去jdk里面包含的源代碼里面自己去親自看看vector和ArrayList的區(qū)別吧。當(dāng)然如果是jdk5.0之后的會(huì)比較難看一點(diǎn),因?yàn)橛旨尤肓朔盒偷恼Z法,類似c+的template語法。 那么大家是否想過為什么要從舊集合框架默認(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)誕生的語言。互聯(lián)網(wǎng)意味著什么?多線程是必然的趨勢(shì)??陀^環(huán)境在變,Java技術(shù)也隨著飛速發(fā)展,導(dǎo)致越來越脫離它的初衷。據(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)無人可比。微軟如何打敗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è)巨人是誰呢?很多專家預(yù)言將在亞洲或者中國出現(xiàn),Whatever,客觀環(huán)境變化趨勢(shì)才是決定大方向的關(guān)鍵。當(dāng)然筆者也希望會(huì)出現(xiàn)在中國,_ 2關(guān)于Java設(shè)計(jì)模式 身邊的很多在看GOF的23種設(shè)計(jì)模式,似乎學(xué)習(xí)它無論在學(xué)校還是在職場(chǎng),都成了一種流行風(fēng)氣。我不想列舉解釋這23種DesignPattern,我寫這些的初衷一直都是談自己的經(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自身的類庫幫助記憶,例如比較常用的觀察者模式,在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ì)模式,作者:StevenJohnMetsker,大部分例子老實(shí)說令現(xiàn)在的我也很迷惑。但我仍然不同意入門跟學(xué)習(xí)設(shè)計(jì)模式有任何沖突,只是我們需要知道每種模式的概念的和典型的應(yīng)用,這樣我們?cè)诘谝淮尉帉慒ileOutputStream、BufferedReader、PrintWriter的時(shí)候就能感覺到原來設(shè)計(jì)模式離我們?nèi)绱酥?,而且并不是多么神秘的東西。 另外,在學(xué)習(xí)某些模式的同時(shí),反而更能幫助我們理解java類庫的某些特點(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ì)象序列化到磁盤上再反序列化回來,這樣不用判斷就可以得到一個(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來代理實(shí)現(xiàn)。簡(jiǎn)單來說就是Adult里的buy()內(nèi)部實(shí)際調(diào)用的是Child的buy(),可是暴露在main函數(shù)的卻是Adult.buy()。這樣一個(gè)簡(jiǎn)單的程序就足夠理解代理模式的基本含義了。Java雜談(三) 這已經(jīng)筆者寫的第三篇Java雜記了,慶幸前兩篇一直得到論壇朋友們的支持鼓勵(lì),還望大家繼續(xù)指正不足之處。筆者也一直渴望通過這樣方式清醒的自審,來尋找自己技術(shù)上的不足之處,希望和共同愛好Java的同仁們一起提高。 前兩次分別講述了關(guān)于jvm、jdk、jre、collection、classLoader和一些DesignPattern的自我理解。這次仍然不準(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)就是用=來比較,即直接比較內(nèi)存地址,返回true或者false。而toString()方法,返回的串組成方式是 getClass().getName()+Integer.toHexString(hashCode() 其實(shí)不用我過多的解釋,大家都能看懂這個(gè)串的組成。接下來再看看hashCode(): publicnativeinthashCode(); 由于是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)致更快的被銷毀來節(jié)約內(nèi)存提高性能。其實(shí)從常理來說,我們也可以大概這樣猜測(cè)出jvm做法的目的。 2關(guān)于重載hashCode()與Collection框架的關(guān)系 筆者曾經(jīng)聽一位搞Java培訓(xùn)多年的前輩說在他看來hashCode方法沒有任何意義,僅僅是為了配合證明具有同樣的hashCode會(huì)導(dǎo)致equals方法相等而存在的。連有的前輩都犯這樣的錯(cuò)誤,其實(shí)說明它還是滿容易被忽略的。那么hashCode()方法到底做什么用? 學(xué)過數(shù)據(jù)結(jié)構(gòu)的課程大家都會(huì)知道有一種結(jié)構(gòu)叫hashtable,目的是通過給每個(gè)對(duì)象分配一個(gè)唯一的索引來提高查詢的效率。那么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來存組合結(jié)果,然后把隨機(jī)產(chǎn)生的組合加入一個(gè)HashSet中,就是想利用HashSet不包括重復(fù)元素的特點(diǎn)。可是HashSet怎么判斷是不是重復(fù)的元素呢?當(dāng)然是通過hashCode()返回的結(jié)果是否相等來判斷啦,可做一下這個(gè)實(shí)驗(yàn): intA=1,2,3,4; intB=1,2,3,4; System.out.println(A.hashCode(); System.out.println(B.hashCode(); 這明明是同一種組合,卻是不同的hashCode,加入Set的時(shí)候會(huì)被當(dāng)成不同的對(duì)象。這個(gè)時(shí)候我們就需要自己來重寫hashCode()方法了,如何寫呢?其實(shí)也是基于原始的hashCode(),畢竟那是操作系統(tǒng)的實(shí)現(xiàn),找到相通對(duì)象唯一的標(biāo)識(shí),實(shí)現(xiàn)方式很多,筆者的實(shí)現(xiàn)方式是: 首先重寫了toString()方法: returnA0“+”A1“+”A2“+”A3;/顯示上比較直觀 然后利用toString()來計(jì)算hashCode(): returnthis.toString().hashCode(); 這樣上述A和B返回的就都是”1234”,在測(cè)試toString().hashCode(),由于String在內(nèi)存中的副本是一樣的,”1234”.hashCode()返回的一定是相同的結(jié)果。 說到這,相信大家能理解得比我更好,今后千萬不要再誤解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類來實(shí)現(xiàn)內(nèi)存反射機(jī)制的。 熟悉C+的人一定知道下面這樣在C+中是做不到的:運(yùn)行時(shí)以字符串參數(shù)傳遞一個(gè)類名,就可以得到這個(gè)類的所有信息,包括它所有的方法,和方法的詳細(xì)信息。還可以實(shí)例化一個(gè)對(duì)象,并通過查到的方法名來調(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)單的例子足以說明問題: classA synchronizedvoidf() voidg() 當(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è)方面來進(jìn)行總結(jié),談?wù)勛约旱慕?jīng)驗(yàn)和想法。希望大家還能一如既往的支持筆者寫下去,指正不足之處。Java雜談(四) 不知不覺已經(jīng)寫到第四篇了,論壇里面不斷的有朋友鼓勵(lì)我寫下去。堅(jiān)持自己的作風(fēng),把一切迷惑不容易理清楚的知識(shí)講出來,講到大家都能聽懂,那么自己就真的懂了。最近在公司實(shí)習(xí)的時(shí)候Trainer跟我講了很多經(jīng)典事跡,對(duì)還未畢業(yè)的我來說是筆不小的財(cái)富,我自己的信念是:人在逆境中成長(zhǎng)的速度要遠(yuǎn)遠(yuǎn)快過順境中,這樣來看一切都能欣然接受了。 好了,閑話不說了,第三篇講的是反射機(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)制流重新組裝成原來的對(duì)象逆過程。它們?cè)贘ava中分別是通過ObjectInputStream和ObjectInputStream這兩個(gè)類來實(shí)現(xiàn)的(以下分別用ois和oos來簡(jiǎn)稱)。 oos的writeObject()方法用來執(zhí)行序列化的過程,ois的readObject()用來執(zhí)行反序列化的過程,在傳輸二進(jìn)制流之前,需要講這兩個(gè)高層流對(duì)象連接到同一個(gè)Channel上,這個(gè)Channel可以是磁盤文件,也可以是socket底層流。所以無論用哪種方式,底層流對(duì)象都是以構(gòu)造函數(shù)參數(shù)的形式傳遞進(jìn)oos和ois這兩個(gè)高層流,連接完畢了才可以進(jìn)行二進(jìn)制數(shù)據(jù)傳輸?shù)?。例子?可以是文件流通道 file=newFile(“C:/data.dat”); oos=newObjectOutputStream(newFileOutputStream(file); ois=newObjectInputStream(newFileInputStream(file); 或者網(wǎng)絡(luò)流通道 oos=newObjectOutputStream(socket.getOutputStream(); ois=newObjectInputStream(socket.getInputStream(); 不知道大家是否注意到oos總是在ois之前定義,這里不希望大家誤解這個(gè)順序是固定的么?回答是否定的,那么有順序要求么?回答是肯定的。原則是什么呢? 原則是互相對(duì)接的輸入/輸出流之間必須是output流先初始化然后再input流初始化,否則就會(huì)拋異常。大家肯定會(huì)問為什么?只要稍微看一看這兩個(gè)類的源代碼文件就大概知道了,output流的任務(wù)很簡(jiǎn)單,只要把對(duì)象轉(zhuǎn)換成二進(jìn)制往通道中寫就可以了,但input流需要做很多準(zhǔn)備工作來接受并最終重組這個(gè)Object,所以O(shè)bjectInputStream的構(gòu)造函數(shù)中就需要用到output初始化發(fā)送過來的header信息,這個(gè)方法叫做readStreamHeader(),它將會(huì)去讀兩個(gè)Short值用于決定用多大的緩存來存放通道發(fā)送過來的二進(jìn)制流,這個(gè)緩存的size因jre的版本不同是不一樣的。所以output如果不先初始化,input的構(gòu)造函數(shù)首先就無法正確運(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è)接連不斷的傳過來,有很多人在運(yùn)行的時(shí)候會(huì)碰到EOFException,然后百思不得其解,去各種論壇問解決方案。其實(shí)筆者這里想說,這個(gè)異常不是必須聲明的,也就是說它雖然是異常,但其實(shí)是正常運(yùn)行結(jié)束的標(biāo)志。EOF表示讀到了文件尾,發(fā)送結(jié)束自然連接也就斷開了。如果這影響到了你程序的正確性的話,請(qǐng)各位靜下心來看看自己程序的業(yè)務(wù)邏輯,而不要把注意力狹隘的聚集在發(fā)送和接受的方法上。因?yàn)楣P者也被這樣的bug困擾了1整天,被很多論壇的帖子誤解了很多次最后得出的教訓(xùn)。如果在while循環(huán)中去readObject,本質(zhì)上是沒有問題的,有對(duì)象數(shù)據(jù)來就會(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)懂了不妨來回答下面的問題: a.A對(duì)象實(shí)現(xiàn)Runnable接口,A.start()運(yùn)行后所謂的線程對(duì)象是誰?是A么? b.線程的wait()、notify()方法到底是做什么時(shí)候用的,什么時(shí)候用? c.為什么線程的suspend方法會(huì)被標(biāo)注過時(shí),不推薦再使用,線程還能掛起么? d.為了同步我們會(huì)對(duì)線程方法聲明Synchronized來加鎖在對(duì)象上,那么如果父類的f()方法加了Synchronized,子類重寫f()方法必須也加Synchronized么?如果子類的f()方法重寫時(shí)聲明Synchronized并調(diào)用super.f(),那么子類對(duì)象上到底有幾把鎖呢?會(huì)因?yàn)楦?jìng)爭(zhēng)產(chǎn)生死鎖么? 呵呵,各位能回答上來幾道呢?如果這些都能答上來,說明對(duì)線程的概念還是滿清晰的,雖說還遠(yuǎn)遠(yuǎn)不能算精通。筆者這里一一做回答,礙于篇幅的原因,筆者盡量說得簡(jiǎn)介一點(diǎn),如果大家有疑惑的歡迎一起討論。 首先第一點(diǎn),線程跟對(duì)象完全是兩回事,雖然我們也常說線程對(duì)象。但當(dāng)你用run()和start()來啟動(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()來讓它們自己根據(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ù)的代碼段,暢通無阻,所以叫重進(jìn)入,我們可以簡(jiǎn)單認(rèn)為第二把鎖沒有加上去。 總而言之,Synchronized的本質(zhì)是不讓其他線程在同一對(duì)象上再加一把鎖。Java雜談(五)本來預(yù)計(jì)J2se只講了第四篇就收尾了,可是版主厚愛把帖子置頂長(zhǎng)期讓大家瀏覽讓小弟倍感責(zé)任重大,務(wù)必追求最到更好,所以關(guān)于J2se一些沒有提到的部分,決定再寫幾篇把常用的部分經(jīng)驗(yàn)全部寫出來供大家討論切磋。這一篇準(zhǔn)備講一講Xml解析包和JavaSwing,然后下一篇再講java.security包關(guān)于Java沙箱安全機(jī)制和RMI機(jī)制,再進(jìn)入J2ee的部分,暫時(shí)就做這樣的計(jì)劃了。如果由于實(shí)習(xí)繁忙更新稍微慢了一些,希望各位見諒! 1Java關(guān)于XML的解析 相信大家對(duì)XML都不陌生,含義是可擴(kuò)展標(biāo)記語言。本身它也就是一個(gè)數(shù)據(jù)的載體以樹狀表現(xiàn)形式出現(xiàn)。后來慢慢的數(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í)候誰也沒想到的。 當(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ì)上并沒有誰好誰不好,只是實(shí)現(xiàn)的思想不一樣罷了。給一個(gè)XML文件的例子: ACat 所謂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ā)為基本思路,順序的搜索下來,碰到了Element之前觸發(fā)什么事件,碰到之后做什么動(dòng)作。由于需要自己來寫觸發(fā)事件的處理方案,所以需要借助另外一個(gè)自定義的Handler,處于org.xml.sax.helpers包中。它的優(yōu)點(diǎn)當(dāng)然是不用整個(gè)包都讀入內(nèi)存,缺點(diǎn)也是只能順序搜索,走完一遍就得重來。 大家很容易就能猜到,接觸到的J2ee框架用的是哪一種,顯然是DOM。因?yàn)轭愃芐truts,Hibernate框架配置文件畢竟是很小的一部分配置信息,而且需要頻繁搜索來讀取,當(dāng)然會(huì)采用DOM方式(其實(shí)SAX內(nèi)部也是用DOM采用的結(jié)構(gòu)來存儲(chǔ)節(jié)點(diǎn)信息的)?,F(xiàn)在無論用什么框架,還真難發(fā)現(xiàn)使用SAX來解析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里全部的接口目的所在。在筆者看來,簡(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的語法中間找到相對(duì)應(yīng)的含義。由于這里不是講解XML基本語法,就不多介紹了。如果大家感興趣,筆者也可以專門寫一篇關(guān)于XML的語法規(guī)則帖與大家分享一下。 2JavaSwing Swing是一個(gè)讓人又愛又恨的東西,可愛之處在于上手很容易,較AWT比起來Swing提供的界面功能更加強(qiáng)大,可恨之處在于編復(fù)雜的界面工作量實(shí)在是巨大。筆者寫過超過3000行的Swing界面,感覺用戶體驗(yàn)還不是那么優(yōu)秀。最近又寫過超過6000行的,由于功能模塊多了,整體效果還只是一般般。體會(huì)最深的就一個(gè)字:累!所以大家現(xiàn)在都陸續(xù)不怎么用Swing在真正開發(fā)的項(xiàng)目上了,太多界面技術(shù)可以取代它了。筆者去寫也是迫于無奈組里面大家都沒寫過,我不入地域誰入? 盡管Swing慢慢的在被人忽略,特別是隨著B/S慢慢的在淹沒C/S,筆者倒是很愿意站出來為Swing正身。每一項(xiàng)技術(shù)的掌握絕不是為了流行時(shí)尚跟風(fēng)。真正喜歡Java的朋友們還是應(yīng)該好好體會(huì)一下Swing,相信在校的很多學(xué)生也很多在學(xué)習(xí)它。很可能從Jdk1.1、1.2走過來的很多大學(xué)老師可能是最不熟悉它的。 Swing提供了一組輕組件統(tǒng)稱為JComponent,它們與AWT組件的最大區(qū)別是JComponent全部都是Container,而Container的特點(diǎn)是里面可以裝載別的組件。在Swing組件中無論是JButton、JLabel、JPanel、JList等都可以再裝入任何其他組件。好處是程序員可以對(duì)Swing組件實(shí)現(xiàn)“再開發(fā)”,針對(duì)特定需求構(gòu)建自己的按鈕、標(biāo)簽、畫板、列表之類的特定組件。 有輕自然就有重,那么輕組件和重組件區(qū)別是?重組件表現(xiàn)出來的形態(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í)出來的Dialog也需要依托一個(gè)Frame窗體,如果沒有指定就會(huì)默認(rèn)產(chǎn)生一個(gè)然后裝載這個(gè)Dialog顯示出來。 b.JFrame是由這么幾部分組成: 最底下一層JRootPane,上面是glassPane(一個(gè)JPanel)和layeredPane(一個(gè)JLayeredPane),而layeredPane又由contentPane(一個(gè)JPanel)和menuBar構(gòu)成。我們的組件都是加在contentPane上,而背景圖片只能加在layeredPane上面。至于glassPane是一個(gè)透明的覆蓋了contentPane的一層,在特定效果中將被利用到來記錄鼠標(biāo)坐標(biāo)或掩飾組件。 c.為了增強(qiáng)用戶體驗(yàn),我們會(huì)在一些按鈕上添加快捷鍵,但Swing里面通常只能識(shí)別鍵盤的Alt鍵,要加入其他的快捷鍵,必須自己實(shí)現(xiàn)一個(gè)ActionListener。 d.通過setLayout(null)可以使得所有組件以setBounds()的四個(gè)參數(shù)來精確定位各自的大小、位置,但不推薦使用,因?yàn)楹玫木幊田L(fēng)格不應(yīng)該在Swing代碼中硬編碼具體數(shù)字,所有的數(shù)字應(yīng)該以常數(shù)的形式統(tǒng)一存在一個(gè)靜態(tài)無實(shí)例資源類文件中。這個(gè)靜態(tài)無實(shí)例類統(tǒng)一負(fù)責(zé)Swing界面的風(fēng)格,包括字體和顏色都應(yīng)該包括進(jìn)去。 e.好的界面設(shè)計(jì)有一條GoldenRule:用戶不用任何手冊(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沙箱安全框架。 1Java中的RMI機(jī)制 RMI的全稱是遠(yuǎn)程方法調(diào)用,相信不少朋友都聽說過,基本的思路可以用一個(gè)經(jīng)典比方來解釋: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ī)端就無法看到B執(zhí)行這段功能的過程和代碼,因?yàn)榭炊伎床坏?,所以既沒有機(jī)會(huì)竊取也沒有機(jī)會(huì)去改動(dòng)方法代碼。EJB正式基于這樣的思想來完成它的任務(wù)的。當(dāng)簡(jiǎn)單的加法變成復(fù)雜的數(shù)據(jù)庫操作和電子商務(wù)交易應(yīng)用的時(shí)候,這樣的安全性和分布式應(yīng)用的便利性就表現(xiàn)出來優(yōu)勢(shì)了。 好了,回到細(xì)節(jié)上,要如何實(shí)現(xiàn)遠(yuǎn)程方法調(diào)用呢?我希望大家學(xué)習(xí)任何技術(shù)的時(shí)候可以試著依賴自己的下意識(shí)判斷,只要你的想法是合理健壯的,那么很可能實(shí)際上它就是這么做的,畢竟真理都蘊(yùn)藏在平凡的生活細(xì)節(jié)中。這樣只要帶著一些薄弱的Java基礎(chǔ)來思考RMI,其實(shí)也可以想出個(gè)大概來。 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è)例子: publicinterfacePe

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。

評(píng)論

0/150

提交評(píng)論