成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

泡圖書館,我想到了享元模式

開發 前端
大家好,我是老田,今天我給大家分享設計模式中的享元模式。用貼切的生活故事,以及真實項目場景來講設計模式,最后用一句話來總結這個設計模式。

[[404702]]

大家好,我是老田,今天我給大家分享設計模式中的享元模式。用貼切的生活故事,以及真實項目場景來講設計模式,最后用一句話來總結這個設計模式。

下面是本文目錄:

背景

享元模式(Flyweight Pattern)又叫作輕量級模式,是對象池的一種實現。

類似線程池,線程池可以避免不停地創建和銷毀多個對象,消耗性能。

享元模式提供了減少對象數量從而改善應用所需的對象結構的方式。

英文解釋:

Use sharing to support large numbers of fine-grained objects efficiently.

享元模式(Flyweight Pattern)其宗旨是共享細粒度對象,將多個對同一對象的訪問集中起來,不必為每個訪問者都創建一個單獨的對象, 主要用于減少創建對象的數量,以減少內存占用和提高性能。

屬于結構性設計模式,其中結構性設計模式有:代理、門面、裝飾器、享元、橋接、適配器、組合。

注意:

享元模式把一個對象的狀態分成內部狀態和外部狀態,內部狀態是不變的,外部狀態是變化的;然后通過共享不變的部分,達到減少對象數量并節約內存的目的。

生活案例

房屋中介

只要是個城市,就少不了房屋中介,房屋中介存有大量的出租房屋信息,并且一家房屋中介往往會有多個門店,但是所有門店都共享這些房屋信息(共享的是出租房屋的信息)。

個人身份證信息

每個中國公民都有一張身份證,并且這張身份證信息在公安系統中是共享的,全國各公安局派出所都會共享你的身份證信息(共享的是個人身份信息)。

高考志愿填報

每所大學在每個省都有明確的招收名額,這些名額對于該省的所有高考生而言都是共享的(共享的是招收名額)。

圖書館

圖書館里的可借書籍,對多有讀者是共享的,大家都可以查詢此書是否已經被借出去,還剩基本可借(共享的是圖書)。

....

簡單代碼實現

下面我們通過一個案例來演示享元模式(圖書館為例)。

  1. public interface Book { 
  2.     void borrow(); 
  3. /** 
  4.  * @author java后端技術全棧 
  5.  */ 
  6. public class ConcreteBook implements Book { 
  7.     //被借出去的書名 
  8.     private String name
  9.  
  10.     public ConcreteBook(String name) { 
  11.         this.name = name
  12.     } 
  13.  
  14.     @Override 
  15.     public void borrow() { 
  16.         System.out.println("圖書館借出去一本書,書名:"+this.name); 
  17.     } 
  18. import java.util.HashMap; 
  19. import java.util.Map; 
  20.  
  21. /** 圖書館 
  22.  * @author java后端技術全棧 
  23.  */ 
  24. public class Llibrary { 
  25.     private Map<String, Book> bookMap = new HashMap<>(); 
  26.  
  27.     private Llibrary() { 
  28.     } 
  29.  
  30.     //只能有一個圖書館 
  31.     public static Llibrary getInstance() { 
  32.         return LazyHolder.LAZY_STATIC_SINGLETON; 
  33.     } 
  34.  
  35.     //通過書名name來借書 
  36.     public Book libToBorrow(String name) { 
  37.         Book book; 
  38.         //如果圖書館有,直接把書借走 
  39.         if (bookMap.containsKey(name)) { 
  40.             book = bookMap.get(name); 
  41.         } else {//圖書館沒有,則錄入一本書,然后把書借走 
  42.             book = new ConcreteBook(name); 
  43.             bookMap.put(name, book); 
  44.         } 
  45.         return book; 
  46.     } 
  47.  
  48.     //返回還有多少本書 
  49.     public int bookSize() { 
  50.         return bookMap.size(); 
  51.     } 
  52.     private static class LazyHolder { 
  53.         private static final Llibrary LAZY_STATIC_SINGLETON = new Llibrary(); 
  54.     } 
  55. import java.util.ArrayList; 
  56. import java.util.List; 
  57.  
  58. public class Student { 
  59.     private static List<Book> bookList = new ArrayList<>(); 
  60.     private static BookFactory bookFactory; 
  61.  
  62.     public static void main(String[] args) { 
  63.         bookFactory = BookFactory.getInstance(); 
  64.  
  65.         studenBorrow("java 從入門到精通"); 
  66.         studenBorrow("java 從入門到放棄"); 
  67.         studenBorrow("JVM java虛擬機"); 
  68.         studenBorrow("java編程思想"); 
  69.  
  70.  
  71.         //還了后,再借一次 
  72.         studenBorrow("java 從入門到精通"); 
  73.         studenBorrow("java 從入門到放棄"); 
  74.         studenBorrow("JVM java虛擬機"); 
  75.         studenBorrow("java編程思想"); 
  76.  
  77.         //還了后,再借一次 
  78.         studenBorrow("java 從入門到精通"); 
  79.         studenBorrow("java 從入門到放棄"); 
  80.         studenBorrow("JVM java虛擬機"); 
  81.         studenBorrow("java編程思想"); 
  82.  
  83.         //把每一本書借出去 
  84.         for (Book book:bookList){ 
  85.             book.borrow(); 
  86.         } 
  87.  
  88.         System.out.println("學生一共借了 "+bookList.size()+"本書"); 
  89.         System.out.println("學生一共借了 "+ bookFactory.bookSize()+"本書"); 
  90.  
  91.     } 
  92.  
  93.     private static void studenBorrow(String name) { 
  94.         bookList.add(bookFactory.libToBorrow(name)); 
  95.     } 

運行結果

  1. 圖書館借出去一本書,書名:java 從入門到精通 
  2. 圖書館借出去一本書,書名:java 從入門到放棄 
  3. 圖書館借出去一本書,書名:JVM java虛擬機 
  4. 圖書館借出去一本書,書名:java編程思想 
  5. 圖書館借出去一本書,書名:java 從入門到精通 
  6. 圖書館借出去一本書,書名:java 從入門到放棄 
  7. 圖書館借出去一本書,書名:JVM java虛擬機 
  8. 圖書館借出去一本書,書名:java編程思想 
  9. 圖書館借出去一本書,書名:java 從入門到精通 
  10. 圖書館借出去一本書,書名:java 從入門到放棄 
  11. 圖書館借出去一本書,書名:JVM java虛擬機 
  12. 圖書館借出去一本書,書名:java編程思想 
  13. 學生一共借了 12本書 
  14. 學生一共借了 4本書 

其實,圖書館只有四本書,但是多個人借,A借來看完了,B再去借,B還了C再去借。

這些書籍就被大家共享了。

享元模式的UML類圖如下:

由上圖可以看到,享元模式主要包含3個角色。

  • 抽象享元角色(Book):享元對象抽象基類或者接口,同時定義出對象的外部狀態和內部狀態的接口或實現。
  • 具體享元角色(ConcreteBook):實現抽象角色定義的業務。該角色的內部狀態處理應該與環境無關,不會出現一個操作改變內部狀態、同時修改了外部狀態的情況。
  • 享元工廠(BookFactory):負責管理享元對象池和創建享元對象。

也許這個例子你還是不太明白,下面我們就用工作中常見的場景來解釋一通。

大佬們是怎樣使用的

關于享元模式,在JDK中大量的使用,比如:String、Integer、Long等類中,都有使用到。

Integer中的享元模式

下面這段代碼輸出什么?

  1. /** 
  2.  * 歡迎關注公眾號:java后端技術全棧 
  3.  * 
  4.  * @author 田維常 
  5.  * @date 2021/06/02 19:30 
  6.  */ 
  7. public class IntegerDemo { 
  8.     public static void main(String[] args) { 
  9.         Integer a = 100; 
  10.         Integer b = Integer.valueOf(100); 
  11.         System.out.println(a == b); 
  12.  
  13.         Integer c = new Integer(1000); 
  14.         Integer d = Integer.valueOf(1000); 
  15.         System.out.println(c == d); 
  16.  
  17.     } 

很多人可能會認為輸出

  1. true 
  2. true 

其實,非也,這里最終輸出的是:

  1. true 
  2. false 

為什么呢?100就可以比較,1000就不能比較了?

其實,在Integer里就用到了享元模式,它就是把-128到127這個范圍的數據緩存起來(放在Integer類型的數組中)。

  1. static final int low = -128; 
  2. public static Integer valueOf(int i) { 
  3.     //high默認是127 
  4.     if (i >= IntegerCache.low && i <= IntegerCache.high) 
  5.         return IntegerCache.cache[i + (-IntegerCache.low)]; 
  6.     return new Integer(i); 

下面進行一個簡要的分析:

關于Integer的緩存,推薦看這篇文章:

這里Integer里的IntegerCache里就用到了享元模式。

關于Integer 推薦:面試官:說說Integer緩存范圍

String中的享元模式

Java中講String類定義為final不能繼承,并且將屬性value也定義為final便是不可變,JVM中字符串一般保存在字符串常量池中,Java會確保一個字符串在常量池中只會有一份拷貝,這個字符串常量池在JDK1.6中位于方法區(永久代)中,而JDK1.7以后,JVM講其從方法區移動到了堆heap中。

下面這段代碼輸出什么?

  1. /** 
  2.  * 歡迎關注公眾號:java后端技術全棧 
  3.  * 
  4.  * @author 田維常 
  5.  * @date 2021/06/03 
  6.  */ 
  7. public class StringDemo { 
  8.     public static void main(String[] args) throws Exception { 
  9.         String s1 = "abcd"
  10.         String s2 = "abcd"
  11.  
  12.  
  13.         String s3 = "ab" + "cd"
  14.         String s4 = "ab" + new String("cd"); 
  15.  
  16.         String s5 = new String("abcd"); 
  17.         String s6 = s5.intern(); 
  18.  
  19.         String s7 = "a"
  20.         String s8 = "bcd"
  21.  
  22.         String s9 = s7 + s8; 
  23.  
  24.         System.out.println("s1 == s2 " + (s1 == s2)); 
  25.  
  26.         System.out.println("s1 == s3 " + (s1 == s3)); 
  27.         System.out.println("s1 == s4 " + (s1 == s4)); 
  28.         System.out.println("s1 == s6 " + (s1 == s6)); 
  29.         System.out.println("s1 == s9 " + (s1 == s9)); 
  30.         System.out.println("s4 == s5 " + (s4 == s5)); 
  31.  
  32.     } 

String類中的value是final修飾的,以字面量的形式創建String變量時,JVM會在編譯期間就把該字面量“abcd”放到字符串常量池匯總,有Java程序啟動的時候就已經加載到內存中了。這個字符串常量的特點就是有且僅有一份相同的字面量,如果其他相同字面量,JVM則返回這個字面量的引用,如果沒有相同的字面量,則再字符串常量池中創建這個字面量并返回它的引用。

由于s2指向字面量"abcd"在常量池中已經存在了(s1先于s2),于是JVM就返回這個字面量綁定的引用,所以s1==s2。

s3中字面量的拼接其實在JVM層已經做了優化,在JVM編譯期間就對s3的拼接做了優化,所以s1、s2、s3都可以理解為是同一個,即s1==s3。

s4中的new String("cd"),此時生成了兩個對象,"cd"和new String("cd"),"cd"存在于字符串常量池中,new String("cd")存在于堆heap中,String s4="ab"+ new String("cd");實質上是兩個對象的相加,編譯器不會對其進行優化,相加的結果存在于堆heap中,而s2存在于字符串常量池中,當然不相等,即s1!=s4。

s4和s5最終的結果都是在堆中,所以此時s4!=s5

s5.intern()方法能是一個維度對總的字符串在運行期間動態地加入到字符串常量池中(字符串常量池的內容是程序啟動的時候就以及酒精加載好了,如果字符串常量池中存在該對象對應的字面量,則返回該字面量在字符串常量池中的引用,否則,創建復制一份該字面量到字符串常量池中并發那會它的引用),因此s1==s6。

s9是s7和s8拼接而成,但是jvm并沒有對其進行優化,所以s1!=s9

最后,上面這段代碼輸出:

  1. s1 == s2 true 
  2. s1 == s3 true 
  3. s1 == s4 false 
  4. s1 == s6 true 
  5. s1 == s9 false 
  6. s4 == s5 false 

JVM中的常量池也是享元模式的經典實現之一。

關于String延伸內容:

美團面試題:String s = new String("111")會創建幾個對象?

Long中的享元模式

Long中和Integer中類似,也是最-128到127的數進行了緩存,請看Long中的valueOf()方法源碼部分:

  1. public static Long valueOf(long l) { 
  2.     final int offset = 128; 
  3.     if (l >= -128 && l <= 127) { // will cache 
  4.         return LongCache.cache[(int)l + offset]; 
  5.     } 
  6.     return new Long(l); 

這個就沒必要進行演示了,和Integer一樣,都是使用了緩存,也就是享元模式。

在Apache Commons Pool中的享元模式

對象池化的基本思路是:將用過的對象保存起來,等下一次需要這種對象的時候,再拿出來重復使用,從而在一定程度上減少頻繁創建對象造成的消耗。用于充當保存對象的“容器”的對象,被稱為對象池(Object Pool,簡稱Pool)。

Apache Pool實現了對象池的功能,定義了對象的生成、銷毀、激活、鈍化等操作及其狀態轉換,并提供幾個默認的對象池實現,

有如下幾個重要的角色:

  • Pooled Object(池化對象):用于封裝對象(例如,線程、數據庫連接和TCP連接),將其包裹成可被對象池管理的對象。
  • Pooled Object Factory(池化對象工廠):定義了操作Pooled Object實例生命周期的一些方法,Pooled Object Factory必須實現線程安全。
  • Object Pool(對象池):Object Pool負責管理Pooled Object,例如,借出對象、返回對象、校驗對象、有多少激活對象和有多少空閑對象。

在ObjectPool類的子類org.apache.commons.pool2.impl.GenericObjectPool種有個屬性:

  1. private final Map<IdentityWrapper<T>, PooledObject<T>> allObjects; 

這個Map就是用來緩存對象的,所以這里也是享元模式的實現。

享元模式的擴展

享元模式中的狀態

享元模式的定義提出了兩個要求:細粒度和共享對象。

因為要求細粒度,所以不可避免地會使對象數量多且性質相近,此時我們就將這些對象的信息分為兩個部分:內部狀態和外部狀態。

內部狀態指對象共享出來的信息,存儲在享元對象內部,并且不會隨環境的改變而改變;

外部狀態指對象得以依賴的一個標記,隨環境的改變而改變,不可共享。

比如:連接池中的連接對象,保存在連接對象中的用戶名、密碼、連接URL等信息,在創建對象的時候就設置好了,不會隨環境的改變而改變,這些為內部狀態。而當每個連接要被回收利用時,我們需要將它標記為可用狀態,這些為外部狀態。

優缺點

優點

  • 減少對象的創建,降低內存中對象的數量,降低系統的內存,提高效率。
  • 減少內存之外的其他資源占用。

缺點

  • 關注內、外部狀態,關注線程安全問題。
  • 使系統、程序的邏輯復雜化。

總結

享元模式,單從概念來講估計很多人不是很理解,但是從Integer、String已經生活中的場景結合起來理解,就能輕松理解享元模式,享元模式的實現基本上都伴隨著一個集合用來存這些對象。

一句話總結:

優化資源配置,減少資源浪費

參考:Tom的設計模式課程

好了,今天的分享就到此結束,希望大家能明白什么是享元模式,享元模式的思想我們在開發中是否能借鑒,面試的時候就不要再說你不會設計模式了。

本文轉載自微信公眾號「Java后端技術全棧」,可以通過以下二維碼關注。轉載本文請聯系Java后端技術全棧公眾號。

 

責任編輯:武曉燕 來源: Java后端技術全棧
相關推薦

2015-05-27 14:24:49

2013-01-23 09:44:15

開源軟件開源技術

2011-11-04 10:05:12

Kindle

2010-01-04 20:16:06

高校圖書館SSL VPNArray

2009-05-05 13:57:33

array應用交付網絡

2013-11-11 16:34:36

2010-06-12 09:54:40

圖書館管理系統中UML

2010-10-26 11:39:48

2017-04-14 11:14:16

智慧圖書館西安

2021-08-30 14:30:38

網絡攻擊黑客網絡安全

2014-12-17 18:17:28

2011-10-27 12:00:19

2017-05-11 09:51:29

新華三

2010-07-06 17:06:31

UML建模

2019-10-21 18:39:31

華為

2011-07-07 17:45:03

掃描儀評測

2013-11-06 15:26:42

初志科技數據動車

2013-11-26 09:53:50

2015-07-13 15:12:00

圖書館浪潮云海OS

2009-06-08 16:31:11

SAG網關遠程接入聯想網御
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 91麻豆精品国产91久久久资源速度 | 色综合99| 日韩精品999 | 波多野结衣中文字幕一区二区三区 | 欧美在线一区二区三区 | 精品国产伦一区二区三区观看方式 | av片免费| 日韩欧美国产不卡 | 欧美v日韩v | 国产在线观看 | 亚洲国产成人久久久 | h视频网站在线观看 | 久久激情视频 | 亚洲精品久 | 综合久久综合久久 | 日韩一级不卡 | 人人干人人爽 | 午夜网站视频 | 91精品国产综合久久久久久丝袜 | 欧美一级二级三级 | 蜜桃视频在线观看免费视频网站www | 成人福利视频网站 | 午夜精品久久久久久久久久久久久 | 久久影音先锋 | 99热视| 日日夜夜精品 | 毛片网在线观看 | 日韩欧美不卡 | 国产精品欧美一区二区三区 | 久久久精品视频一区二区三区 | 日韩欧美日韩在线 | 国产精品视频在线播放 | 久久蜜桃av | 久久一本 | 成人18亚洲xxoo | 日韩中文字幕在线播放 | 中文字幕一级 | 在线中文字幕视频 | 国产黄色免费网站 | 影音先锋成人资源 | 国产精品视频免费看 |