Java多線程編程之不可變對(duì)象模式

???????在多線程環(huán)境中铁蹈,為了保證共享數(shù)據(jù)的一致性乡括,往往需要對(duì)共享數(shù)據(jù)的使用進(jìn)行加鎖悲雳,但是加鎖操作本身就會(huì)帶來(lái)一定的開銷蚀浆,這里可以使用將共享數(shù)據(jù)使用不可變對(duì)象進(jìn)行封裝映之,從而避免加鎖操作。

1. 模型角色

???????不可變對(duì)象指的是蜡坊,對(duì)象內(nèi)部沒有提供任何可供修改對(duì)象數(shù)據(jù)的方法杠输,如果需要修改共享變量的任何數(shù)據(jù),都需要先構(gòu)建整個(gè)共享對(duì)象秕衙,然后對(duì)共享對(duì)象進(jìn)行整體的替換蠢甲,通過這種方式來(lái)達(dá)到對(duì)共享對(duì)象數(shù)據(jù)一致性的保證。如下是不可變對(duì)象設(shè)計(jì)的類圖:

不可變對(duì)象類圖

如下是各個(gè)角色功能的描述:

  • ImmutableObject:不可變對(duì)象的載體据忘。對(duì)于需要一致性更改的數(shù)據(jù)鹦牛,都需要放入不可變對(duì)象中,對(duì)于不可變對(duì)象勇吊,需要注意如下幾點(diǎn):
    • 不可變對(duì)象的屬性必須使用final修飾曼追,以防止屬性被意外修改,并且final還可以保證JVM在該對(duì)象構(gòu)造完成時(shí)該屬性已經(jīng)初始化成功(JVM在構(gòu)造完對(duì)象時(shí)可能只是為其分配了引用空間汉规,而各個(gè)屬性值可能還未初始化完成)礼殊;
    • 屬性的設(shè)值必須在構(gòu)造方法中統(tǒng)一構(gòu)造完成,其余的方法只是提供的查詢各個(gè)屬性相關(guān)的方法针史;
    • 對(duì)于可變狀態(tài)的引用類型屬性晶伦,如集合,在獲取該類型的屬性時(shí)啄枕,必須返回該屬性的一個(gè)深度復(fù)制結(jié)果婚陪,以防止不可變對(duì)象的該屬性值被客戶端修改;
    • 不可變對(duì)象的類必須使用final修飾频祝,以防止子類對(duì)其本身或其方法進(jìn)行修改泌参;
  • Manipulator:聚合對(duì)象的管理類(某些情況可不用)脆淹。對(duì)于聚合對(duì)象的管理,主要有兩部分:查詢和修改沽一。對(duì)于聚合對(duì)象的查詢未辆,只需要根據(jù)一定的規(guī)則在Manipulator類中獲取該對(duì)象即可,對(duì)于聚合對(duì)象的修改锯玛,需要首先通過參數(shù)構(gòu)造一個(gè)完整的聚合對(duì)象,然后將保存的該聚合對(duì)象的引用進(jìn)行替換即可兼蜈;
  • Client:獲取聚合對(duì)象的客戶端應(yīng)用攘残。

2. 使用場(chǎng)景

???????對(duì)于不可變對(duì)象,其主要有如下三種使用場(chǎng)景:

  • 當(dāng)某組數(shù)據(jù)變化不是很頻繁为狸,則可以使用不可變對(duì)象歼郭。對(duì)于數(shù)據(jù)的訪問,由于不可變對(duì)象的引用空間不會(huì)發(fā)生變化辐棒,因而任何線程都可以保有同一個(gè)不可變對(duì)象的引用病曾,這樣可以減少內(nèi)存的消耗,也能保證數(shù)據(jù)訪問的一致性漾根;
  • 當(dāng)某組數(shù)據(jù)需要進(jìn)行一致性的更新操作時(shí)泰涂,可以使用不可變對(duì)象。由于不可變對(duì)象能夠保證對(duì)其任何數(shù)據(jù)的修改都是對(duì)整個(gè)對(duì)象的替換辐怕,因而其能夠保證整組數(shù)據(jù)的一致性逼蒙。需要注意的是,如果該組數(shù)據(jù)變更比較頻繁寄疏,則不宜使用不可變對(duì)象是牢,因?yàn)檫@會(huì)造成創(chuàng)建大量的不可變對(duì)象,從而增加JVM垃圾回收的壓力陕截。具體的情況應(yīng)根據(jù)JVM可使用內(nèi)存大小與對(duì)象更新的頻率進(jìn)行考量驳棱;
  • 當(dāng)需要對(duì)象作為Map的鍵時(shí)可以使用不可變對(duì)象。對(duì)于Map而言农曲,其hashCode()方法默認(rèn)返回的是對(duì)象的引用地址社搅,而對(duì)于不可變對(duì)象而言,由于其引用地址是不會(huì)發(fā)生變化的乳规,因而即使不對(duì)其hashCode()方法進(jìn)行重寫罚渐,其也不會(huì)發(fā)生變化。

3. 使用示例

???????對(duì)于不可變對(duì)象驯妄,一個(gè)很好的例子就是地址經(jīng)緯度荷并。筆者所工作的公司處理的業(yè)務(wù)和房源相關(guān),其中有一部分就是需要處理房源所在點(diǎn)的經(jīng)緯度信息青扔,這里就可以使用不可變對(duì)象源织,因?yàn)榉吭唇?jīng)緯度基本上不會(huì)發(fā)生變化翩伪,并且對(duì)其操作也主要是以查詢?yōu)橹鳎钪匾氖翘赶ⅲ瑢?duì)經(jīng)緯度的處理必須是經(jīng)度和緯度同時(shí)發(fā)生變化缘屹,任何情況下只更改了其中一個(gè)數(shù)據(jù)都會(huì)產(chǎn)生問題。如下是記錄房源經(jīng)緯度的類:

public final class Location {
  private final long id;
  private final String latitude;
  private final String longitude;

  public Location(long id, String latitude, String longitude) {
    this.id = id;
    this.latitude = latitude;
    this.longitude = longitude;
  }

  public long getId() {
    return id;
  }

  public String getLatitude() {
    return latitude;
  }

  public String getLongitude() {
    return longitude;
  }
}

???????該Location類也即上述UML類圖中的ImmutableObject部分侠仇∏嶙耍可以看到,任何對(duì)Location對(duì)象的修改都必須重新構(gòu)建一個(gè)Location對(duì)象逻炊。如下是對(duì)Location的管理類互亮,用于存儲(chǔ)具體的Location信息的:

public class LocationHolder {
  
  private final LocationHolder INSTANCE = new LocationHolder();
  private Map<Long, Location> locations;

  private LocationHolder() {
    this.locations = new ConcurrentHashMap<>();
  }
  
  public LocationHolder getInstance() {
    return INSTANCE;
  }

  public Location getLocation(long id) {
    return locations.get(id);
  }

  public void addLocation(long id, String latitude, String longitude) {
    Location location = new Location(id, latitude, longitude);
    locations.put(id, location);
  }
  
  public Map<Long, Location> getLocations() {
    return Collections.unmodifiableMap(locations);
  }
}

????????可以看到,這里對(duì)Location的管理是通過一個(gè)單例類LocationHolder進(jìn)行的余素,任何對(duì)Location的操作都進(jìn)行了封裝豹休,并且這里批量獲取Location,也是返回了一個(gè)不可變Map桨吊,從而保證原始數(shù)據(jù)不會(huì)作任何修改威根,如果該Map的鍵或值任何一方可能發(fā)生變化,那么在返回值則必須返回一個(gè)深度復(fù)制的結(jié)果视乐,這樣才能保證原始數(shù)據(jù)的完整性洛搀。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市佑淀,隨后出現(xiàn)的幾起案子姥卢,更是在濱河造成了極大的恐慌,老刑警劉巖渣聚,帶你破解...
    沈念sama閱讀 216,402評(píng)論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件独榴,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡奕枝,警方通過查閱死者的電腦和手機(jī)棺榔,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)隘道,“玉大人症歇,你說(shuō)我怎么就攤上這事√饭#” “怎么了忘晤?”我有些...
    開封第一講書人閱讀 162,483評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)激捏。 經(jīng)常有香客問我设塔,道長(zhǎng),這世上最難降的妖魔是什么远舅? 我笑而不...
    開封第一講書人閱讀 58,165評(píng)論 1 292
  • 正文 為了忘掉前任闰蛔,我火速辦了婚禮痕钢,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘序六。我一直安慰自己任连,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,176評(píng)論 6 388
  • 文/花漫 我一把揭開白布例诀。 她就那樣靜靜地躺著随抠,像睡著了一般。 火紅的嫁衣襯著肌膚如雪繁涂。 梳的紋絲不亂的頭發(fā)上拱她,一...
    開封第一講書人閱讀 51,146評(píng)論 1 297
  • 那天,我揣著相機(jī)與錄音爆土,去河邊找鬼。 笑死诸蚕,一個(gè)胖子當(dāng)著我的面吹牛步势,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播背犯,決...
    沈念sama閱讀 40,032評(píng)論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼坏瘩,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了漠魏?” 一聲冷哼從身側(cè)響起倔矾,我...
    開封第一講書人閱讀 38,896評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎柱锹,沒想到半個(gè)月后哪自,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,311評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡禁熏,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,536評(píng)論 2 332
  • 正文 我和宋清朗相戀三年壤巷,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片瞧毙。...
    茶點(diǎn)故事閱讀 39,696評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡胧华,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出宙彪,到底是詐尸還是另有隱情矩动,我是刑警寧澤,帶...
    沈念sama閱讀 35,413評(píng)論 5 343
  • 正文 年R本政府宣布释漆,位于F島的核電站悲没,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏男图。R本人自食惡果不足惜檀训,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,008評(píng)論 3 325
  • 文/蒙蒙 一柑潦、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧峻凫,春花似錦渗鬼、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至命锄,卻和暖如春堰乔,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背脐恩。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工镐侯, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人驶冒。 一個(gè)月前我還...
    沈念sama閱讀 47,698評(píng)論 2 368
  • 正文 我出身青樓苟翻,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親骗污。 傳聞我的和親對(duì)象是個(gè)殘疾皇子崇猫,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,592評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容

  • 國(guó)家電網(wǎng)公司企業(yè)標(biāo)準(zhǔn)(Q/GDW)- 面向?qū)ο蟮挠秒娦畔?shù)據(jù)交換協(xié)議 - 報(bào)批稿:20170802 前言: 排版 ...
    庭說(shuō)閱讀 10,958評(píng)論 6 13
  • 1.ios高性能編程 (1).內(nèi)層 最小的內(nèi)層平均值和峰值(2).耗電量 高效的算法和數(shù)據(jù)結(jié)構(gòu)(3).初始化時(shí)...
    歐辰_OSR閱讀 29,372評(píng)論 8 265
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)需忿,斷路器诅炉,智...
    卡卡羅2017閱讀 134,651評(píng)論 18 139
  • 其實(shí)想要重拾起寫日記的習(xí)慣已經(jīng)有段時(shí)間了呵曹,只是現(xiàn)在越來(lái)越眼高手低亚皂,以至于遲遲至今惜犀。先試試這個(gè)平臺(tái)的感覺柔袁,太久不寫了...
    是夜無(wú)眠閱讀 204評(píng)論 0 0
  • “都說(shuō)婚姻是女人的二度投胎吵瞻,我想既然不能選擇自己的出生运嗜,那么對(duì)于婚姻仲翎,我沒理由拒絕一個(gè)有進(jìn)取心的男人……”屏幕面前...
    Y小姐_閱讀 615評(píng)論 2 4