Java mutable對(duì)象和immutable對(duì)象的區(qū)別說明
今天讀jdk源碼中Map.java時(shí)看到一句話:
great care must be exercised if mutable objects are used as map keys;
第一次知道m(xù)utable對(duì)象這個(gè)概念,google了一下,維基百科定義如下:
“In object-oriented and functional programming, an immutable object (unchangeable[1] object) is an object whose state cannot be modified after it is created.[2] This is in contrast to a mutable object (changeable object) , which can be modified after it is created. In some cases, an object is considered immutable even if some internally used attributes change but the object’s state appears to be unchanging from an external point of view. For example, an object that uses memoization to cache the results of expensive computations could still be considered an immutable object.”
在面向?qū)ο蠛秃瘮?shù)式編程中,一個(gè)immutable對(duì)象(不可變對(duì)象)是指一旦創(chuàng)建之后狀態(tài)不可改變的對(duì)象。
mutable對(duì)象(可變對(duì)象)是指創(chuàng)建之后也可以修改的對(duì)象。
在有些情況下,對(duì)象也被認(rèn)為是不可變的(immutable),即,一個(gè)對(duì)象包含的內(nèi)部使用的屬性改變了,但從外部看對(duì)象的狀態(tài)并沒有改變。
例如,一個(gè)使用memoization來緩存復(fù)雜計(jì)算結(jié)果的對(duì)象仍然被看作是不可變(immutable)對(duì)象.
在面向?qū)ο缶幊讨校琒tring 以及其他的具體對(duì)象都被看作是不可變(immutable)對(duì)象,以提高可讀性和運(yùn)行效率。
不可變對(duì)象有幾個(gè)優(yōu)點(diǎn):線程安全
易于理解
比可變對(duì)象有更高的安全性
Java中不可變對(duì)象的經(jīng)典例子就是String類的實(shí)例:String s = 'ABC';s.toLowerCase();
toLowerCase()方法不會(huì)改變s中包含的數(shù)據(jù)“ABC”。而是創(chuàng)建一個(gè)新的String對(duì)象并將其初始化為“abc”,然后返回這個(gè)新對(duì)象的引用。
盡管String類聲明中沒有提供讓它成為不可變對(duì)象的語法,但是,String類的方法中沒有方法去改變一個(gè)String包含的數(shù)據(jù),這就使得它是不可變的。
Java中關(guān)鍵字final用于聲明原始數(shù)據(jù)類型(primitive types)和對(duì)象引用為不可變對(duì)象,但是它不能使對(duì)象本身變?yōu)椴豢勺儗?duì)象。
原始數(shù)據(jù)類型(primitive types)變量(int, long, short等)定義之后還可以再重新賦值,可以使用final阻止這樣的賦值。
int i = 42; //int is of primitive typei = 43; // OKfinal int j = 42;j = 43; // does not compile. j is final so can’t be reassigned
僅僅使用final關(guān)鍵字還不能讓引用類型(reference types)成為不可變對(duì)象,final只能阻止重新賦值。
final MyObject m = new MyObject(); //m is of reference typem.data = 100; // OK. We can change state of object m (m is mutable and final doesn’t change this fact)m = new MyObject(); // does not compile. m is final so can’t be reassigned
原始類型包裝類(primitive wrappers)(Integer,Long, Short, Double, Float, Character, Byte, Boolean)也都是不可變的。
Java mutable 和 immutable類型含義解釋
immutable : variables that are assigned once and never reassigned.
mutable : When you assign to a variable or a field, you’re changing where the variable’s arrow points. You can point it to a different value. When you assign to the contents of a mutable value ? such as an array or list ? you’re changing references inside that value.
基本類型及其封裝對(duì)象類型都是不可變的
圖形化解釋 Snapshot Diagrammutable:
immutable:
例如String和StringBuilder:
1. String是immutable的,每次對(duì)于String對(duì)象的修改都將產(chǎn)生一個(gè)新的String對(duì)象,而原來的對(duì)象保持不變。
2. StringBuilder是mutable,因?yàn)槊看螌?duì)于它的對(duì)象的修改都作用于該對(duì)象本身,并沒有產(chǎn)生新的對(duì)象。
如何保證自己創(chuàng)建的類是immutable類 所有成員都是private final。 不提供對(duì)成員的改變方法,setXX 確保所有的方法不會(huì)被重寫。手段有兩種:使用final Class(強(qiáng)不可變類),或者將所有類方法加上final(弱不可變類)。 如果某一個(gè)類成員不是原始變量(例如int,double)或者不可變類,必須通過在成員初始化或者使用get方法時(shí)要通過深度拷貝方法,來確保類的不可變。優(yōu)缺點(diǎn)使用不可變類型,對(duì)其頻繁修改會(huì)產(chǎn)生大量的臨時(shí)拷貝(需要垃圾回收) ;
可變類型最 少化拷貝以提高效率。
使用可變數(shù)據(jù)類型,可獲得更好的性能 ,也適合于在多個(gè)模塊之間共享數(shù)據(jù) 。例如全局變量。
不可變類型更“安全”, 在其他質(zhì)量指標(biāo)上表現(xiàn)更好。
對(duì)可變類型可能造成的風(fēng)險(xiǎn),我們通過防御式拷貝(深度拷貝),給客戶端返回一個(gè)全新的可變類型的對(duì)象,大部分時(shí)候該拷貝不會(huì)被客戶端修改, 可能造成大量的內(nèi)存浪費(fèi)。
深度拷貝當(dāng)只是引用傳遞或者根據(jù)對(duì)象的值創(chuàng)建新的值,稱為“淺復(fù)制”,當(dāng)原對(duì)象的發(fā)生改變時(shí),根據(jù)上面方式創(chuàng)建的新對(duì)象的也會(huì)隨之改變;
而如果采用深度復(fù)制,那是真正的復(fù)制了一份新的對(duì)象,新對(duì)象的與原對(duì)象不存在任何關(guān)聯(lián),原對(duì)象發(fā)生改變不會(huì)影響新對(duì)象。
相關(guān)文章:
1. 淺談SpringMVC jsp前臺(tái)獲取參數(shù)的方式 EL表達(dá)式2. 如何在jsp界面中插入圖片3. jsp實(shí)現(xiàn)登錄界面4. 怎樣才能用js生成xmldom對(duì)象,并且在firefox中也實(shí)現(xiàn)xml數(shù)據(jù)島?5. 基于javaweb+jsp實(shí)現(xiàn)企業(yè)車輛管理系統(tǒng)6. ASP.Net MVC利用NPOI導(dǎo)入導(dǎo)出Excel的示例代碼7. 利用ajax+php實(shí)現(xiàn)商品價(jià)格計(jì)算8. jstl 字符串處理函數(shù)9. Android通過Java sdk的方式接入OpenCv的方法10. python爬取新聞門戶網(wǎng)站的示例
