“發(fā)布(Publish)”一個(gè)對(duì)象的意思是指祠锣,使對(duì)象能夠在當(dāng)前作用域之外的代碼中使用
當(dāng)某個(gè)不應(yīng)該發(fā)布的對(duì)象被發(fā)布時(shí),這種情況稱之為“逸出(Escape)”
發(fā)布就是把對(duì)象的引用傳到在當(dāng)前作用域之外涤久,發(fā)布是基于JAVA的引用傳遞機(jī)制,所以這里指的是對(duì)象剥哑,不包括基本類型蚕涤,發(fā)布本身是一種技術(shù)或者機(jī)制,并無(wú)好壞之分纸肉。不“希望”發(fā)布的對(duì)象卻被發(fā)布了溺欧,那么就說(shuō)這個(gè)對(duì)象逸出了。所以說(shuō)這種現(xiàn)象是根據(jù)實(shí)際情況決定的柏肪,如果一個(gè)對(duì)象你“希望”他發(fā)布且發(fā)布了姐刁,那么就是正常發(fā)布,如果一個(gè)對(duì)象你不“希望”被發(fā)布卻發(fā)布了烦味,那么就是逸出龙填。針對(duì)逸出情況,可能會(huì)對(duì)程序運(yùn)行造成影響,特別是多線程處理岩遗,也有可能并不影響程序運(yùn)行。
下面我們看下幾種發(fā)布對(duì)象的方式
- 將一個(gè)指向該對(duì)象的引用保存到其他代碼可以訪問(wèn)的地方--方法體內(nèi)的對(duì)象引用保存到類的公有靜態(tài)變量
例子中發(fā)布的對(duì)象為set凤瘦,如果set添加元素宿礁,則這些元素也被發(fā)布。在這里發(fā)布的意思就是在當(dāng)前作用域中(intitialize方法體中)set對(duì)象的引用保存在knownSecret中蔬芥,這樣就可以在當(dāng)前作用域外通過(guò)knownSecret引用操縱set對(duì)象梆靖,包括set中的元素。
- 將一個(gè)指向該對(duì)象的引用保存到其他代碼可以訪問(wèn)的地方--類的私有對(duì)象的引用通過(guò)公有方法返回
例子中states對(duì)象本來(lái)只作用于類內(nèi)笔诵,但是現(xiàn)在其他對(duì)象可以通過(guò)getStates方法得到states的引用并修改狀態(tài)返吻。
- 將一個(gè)指向該對(duì)象的引用保存到其他代碼可以訪問(wèn)的地方--類對(duì)象傳遞給外部方法
例子中描述的Study類的studyself方法中,因調(diào)用了Enjoy類的watchTV方法導(dǎo)致自身person數(shù)據(jù)狀態(tài)發(fā)生改變乎婿。
以上三種方式都是把對(duì)象的引用發(fā)布到外部测僵,導(dǎo)致外部在持有對(duì)象引用時(shí)可以修改數(shù)據(jù)狀態(tài)。
以下兩種發(fā)布方式發(fā)布的對(duì)象為this谢翎。
- 內(nèi)部類的方式
person是Outer的內(nèi)部屬性捍靠,但是其他類可以通過(guò)內(nèi)部類Inner去訪問(wèn)修改。只要是Outer的屬性森逮,即可通過(guò)this訪問(wèn)的屬性都發(fā)布出去了榨婆。
- 構(gòu)造時(shí)發(fā)布this引用
輸出結(jié)果:
構(gòu)造前:Person [age=0, name=buff]
構(gòu)造后:Person [age=10, name=oba]
線程修改后::Person [age=10, name=oba]
這種情況主要是在對(duì)象還沒(méi)有構(gòu)造完成就對(duì)其引用,并修改了數(shù)據(jù)狀態(tài)。
使用封裝的主要原因:封裝能夠使得對(duì)程序的正確性進(jìn)行分析變得可能褒侧,并使得無(wú)意中破壞設(shè)計(jì)約束條件變得更難良风。
閱讀自此,有種豁然開(kāi)朗的感覺(jué)闷供,才知道在設(shè)計(jì)類時(shí)對(duì)于權(quán)限修飾符的選擇是以此為出發(fā)點(diǎn)的的烟央。以上的發(fā)布情況,都可以通過(guò)權(quán)限修飾符進(jìn)行合理的處理这吻,但這樣的話就無(wú)法進(jìn)行正常的發(fā)布吊档,無(wú)法實(shí)現(xiàn)線程間的數(shù)據(jù)共享。所以本書后面的章節(jié)會(huì)提出解決方式唾糯。