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

Java關(guān)于延遲加載的一些應(yīng)用實(shí)踐

開發(fā) 后端
在Java8中引入的lambda對(duì)于我們實(shí)現(xiàn)延遲操作提供很大的便捷性,如Stream、Supplier等,下面介紹幾個(gè)例子。

[[381098]]

 代碼中的很多操作都是Eager的,比如在發(fā)生方法調(diào)用的時(shí)候,參數(shù)會(huì)立即被求值。總體而言,使用Eager方式讓編碼本身更加簡(jiǎn)單,然而使用Lazy的方式通常而言,即意味著更好的效率。

延遲初始化

一般有幾種延遲初始化的場(chǎng)景:

  •  對(duì)于會(huì)消耗較多資源的對(duì)象:這不僅能夠節(jié)省一些資源,同時(shí)也能夠加快對(duì)象的創(chuàng)建速度,從而從整體上提升性能。
  •  某些數(shù)據(jù)在啟動(dòng)時(shí)無法獲取:比如一些上下文信息可能在其他攔截器或處理中才能被設(shè)置,導(dǎo)致當(dāng)前bean在加載的時(shí)候可能獲取不到對(duì)應(yīng)的變量的值,使用 延遲初始化可以在真正調(diào)用的時(shí)候去獲取,通過延遲來保證數(shù)據(jù)的有效性。

在Java8中引入的lambda對(duì)于我們實(shí)現(xiàn)延遲操作提供很大的便捷性,如Stream、Supplier等,下面介紹幾個(gè)例子。

Lambda

Supplier

通過調(diào)用get()方法來實(shí)現(xiàn)具體對(duì)象的計(jì)算和生成并返回,而不是在定義Supplier的時(shí)候計(jì)算,從而達(dá)到了_延遲初始化_的目的。但是在使用 中往往需要考慮并發(fā)的問題,即防止多次被實(shí)例化,就像Spring的@Lazy注解一樣。 

  1. public class Holder {  
  2.     // 默認(rèn)第一次調(diào)用heavy.get()時(shí)觸發(fā)的同步方法  
  3.     private Supplier<Heavy> heavy = () -> createAndCacheHeavy();   
  4.     public Holder() {  
  5.         System.out.println("Holder created");  
  6.     }  
  7.     public Heavy getHeavy() {  
  8.         // 第一次調(diào)用后heavy已經(jīng)指向了新的instance,所以后續(xù)不再執(zhí)行synchronized  
  9.         return heavy.get();   
  10.     } 
  11.     //...  
  12.     private synchronized Heavy createAndCacheHeavy() {  
  13.         // 方法內(nèi)定義class,注意和類內(nèi)的嵌套class在加載時(shí)的區(qū)別  
  14.         class HeavyFactory implements Supplier<Heavy> {  
  15.             // 饑渴初始化  
  16.             private final Heavy heavyInstance = new Heavy();   
  17.             public Heavy get() {  
  18.                 // 每次返回固定的值 
  19.                 return heavyInstance;   
  20.             }   
  21.         }       
  22.         //第一次調(diào)用方法來會(huì)將heavy重定向到新的Supplier實(shí)例  
  23.         if(!HeavyFactory.class.isInstance(heavy)) {  
  24.             heavy = new HeavyFactory();  
  25.         }  
  26.         return heavy.get();  
  27.     }  

當(dāng)Holder的實(shí)例被創(chuàng)建時(shí),其中的Heavy實(shí)例還沒有被創(chuàng)建。下面我們假設(shè)有三個(gè)線程會(huì)調(diào)用getHeavy方法,其中前兩個(gè)線程會(huì)同時(shí)調(diào)用,而第三個(gè)線程會(huì)在稍晚的時(shí)候調(diào)用。

當(dāng)前兩個(gè)線程調(diào)用該方法的時(shí)候,都會(huì)調(diào)用到createAndCacheHeavy方法,由于這個(gè)方法是同步的。因此第一個(gè)線程進(jìn)入方法體,第二個(gè)線程開始等待。在方法體中會(huì)首先判斷當(dāng)前的heavy是否是HeavyInstance的一個(gè)實(shí)例。

如果不是,就會(huì)將heavy對(duì)象替換成HeavyFactory類型的實(shí)例。顯然,第一個(gè)線程執(zhí)行判斷的時(shí)候,heavy對(duì)象還只是一個(gè)Supplier的實(shí)例,所以heavy會(huì)被替換成為HeavyFactory的實(shí)例,此時(shí)heavy實(shí)例會(huì)被真正的實(shí)例化。

等到第二個(gè)線程進(jìn)入執(zhí)行該方法時(shí),heavy已經(jīng)是HeavyFactory的一個(gè)實(shí)例了,所以會(huì)立即返回(即heavyInstance)。當(dāng)?shù)谌齻€(gè)線程執(zhí)行g(shù)etHeavy方法時(shí),由于此時(shí)的heavy對(duì)象已經(jīng)是HeavyFactory的實(shí)例了,因此它會(huì)直接返回需要的實(shí)例(即heavyInstance),和同步方法createAndCacheHeavy沒有任何關(guān)系了。

以上代碼實(shí)際上實(shí)現(xiàn)了一個(gè)輕量級(jí)的虛擬代理模式(Virtual Proxy Pattern)。保證了懶加載在各種環(huán)境下的正確性。

還有一種基于delegate的實(shí)現(xiàn)方式更好理解一些:

https://gist.github.com/taichi/6daf50919ff276aae74f 

  1. import java.util.concurrent.ConcurrentHashMap;  
  2. import java.util.concurrent.ConcurrentMap;  
  3. import java.util.function.Supplier;  
  4. public class MemoizeSupplier<T> implements Supplier<T> {  
  5.  final Supplier<T> delegate;  
  6.  ConcurrentMap<Class<?>, T> map = new ConcurrentHashMap<>(1);  
  7.  public MemoizeSupplier(Supplier<T> delegate) {  
  8.   this.delegate = delegate;  
  9.  }  
  10.  @Override  
  11.  public T get() {  
  12.      // 利用computeIfAbsent方法的特性,保證只會(huì)在key不存在的時(shí)候調(diào)用一次實(shí)例化方法,進(jìn)而實(shí)現(xiàn)單例  
  13.   return this.map.computeIfAbsent(MemoizeSupplier.class,  
  14.     k -> this.delegate.get());  
  15.  }  
  16.  public static <T> Supplier<T> of(Supplier<T> provider) {  
  17.   return new MemoizeSupplier<>(provider);  
  18.  }  

以及一個(gè)更復(fù)雜但功能更多的CloseableSupplier: 

  1. public static class CloseableSupplier<T> implements Supplier<T>, Serializable {  
  2.         private static final long serialVersionUID = 0L 
  3.         private final Supplier<T> delegate;  
  4.         private final boolean resetAfterClose; 
  5.         private volatile transient boolean initialized;  
  6.         private transient T value;  
  7.         private CloseableSupplier(Supplier<T> delegate, boolean resetAfterClose) {  
  8.             this.delegate = delegate;  
  9.             this.resetAfterClose = resetAfterClose;  
  10.         }  
  11.         public T get() {  
  12.             // 經(jīng)典Singleton實(shí)現(xiàn)  
  13.             if (!(this.initialized)) { // 注意是volatile修飾的,保證happens-before,t一定實(shí)例化完全  
  14.                 synchronized (this) {  
  15.                     if (!(this.initialized)) { // Double Lock Check  
  16.                         T t = this.delegate.get();  
  17.                         tthis.value = t;  
  18.                         this.initialized = true 
  19.                         return t;  
  20.                     }  
  21.                 }  
  22.             }  
  23.             // 初始化后就直接讀取值,不再同步搶鎖  
  24.             return this.value;  
  25.         } 
  26.         public boolean isInitialized() {  
  27.             return initialized;  
  28.         }  
  29.         public <X extends Throwable> void ifPresent(ThrowableConsumer<T, X> consumer) throws X {  
  30.             synchronized (this) {  
  31.                 if (initialized && this.value != null) {  
  32.                     consumer.accept(this.value);  
  33.                 }  
  34.             }  
  35.         } 
  36.         public <U> Optional<U> map(Function<? super T, ? extends U> mapper) {  
  37.             checkNotNull(mapper);  
  38.             synchronized (this) {  
  39.                 if (initialized && this.value != null) {  
  40.                     return ofNullable(mapper.apply(value));  
  41.                 } else {  
  42.                     return empty();  
  43.                 }  
  44.             }  
  45.         }  
  46.         public void tryClose() {  
  47.             tryClose(i -> { });  
  48.         }  
  49.         public <X extends Throwable> void tryClose(ThrowableConsumer<T, X> close) throws X {  
  50.             synchronized (this) {  
  51.                 if (initialized) {  
  52.                     close.accept(value);  
  53.                     if (resetAfterClose) {  
  54.                         this.value = null 
  55.                         initialized = false 
  56.                     }  
  57.                 }  
  58.             }  
  59.         }  
  60.         public String toString() {  
  61.             if (initialized) {  
  62.                 return "MoreSuppliers.lazy(" + get() + ")";  
  63.             } else {  
  64.                 return "MoreSuppliers.lazy(" + this.delegate + ")";  
  65.             }  
  66.         }  
  67.     } 

Stream

Stream中的各種方法分為兩類:

  •  中間方法(limit()/iterate()/filter()/map())
  •  結(jié)束方法(collect()/findFirst()/findAny()/count())

前者的調(diào)用并不會(huì)立即執(zhí)行,只有結(jié)束方法被調(diào)用后才會(huì)依次從前往后觸發(fā)整個(gè)調(diào)用鏈條。但是需要注意,對(duì)于集合來說,是每一個(gè)元素依次按照處理鏈條執(zhí)行到尾,而不是每一個(gè)中間方法都將所有能處理的元素全部處理一遍才觸發(fā) 下一個(gè)中間方法。比如: 

  1. List<String> names = Arrays.asList("Brad", "Kate", "Kim", "Jack", "Joe", "Mike");  
  2. final String firstNameWith3Letters = names.stream()  
  3.     .filter(name -> length(name) == 3)  
  4.     .map(name -> toUpper(name))  
  5.     .findFirst()  
  6.     .get();  
  7. System.out.println(firstNameWith3Letters); 

當(dāng)觸發(fā)findFirst()這一結(jié)束方法的時(shí)候才會(huì)觸發(fā)整個(gè)Stream鏈條,每個(gè)元素依次經(jīng)過filter()->map()->findFirst()后返回。所以filter()先處理第一個(gè)和第二個(gè)后不符合條件,繼續(xù)處理第三個(gè)符合條件,再觸發(fā)map()方法,最后將轉(zhuǎn)換的結(jié)果返回給findFirst()。所以filter()觸發(fā)了_3_次,map()觸發(fā)了_1_次。

好,讓我們來看一個(gè)實(shí)際問題,關(guān)于無限集合。

Stream類型的一個(gè)特點(diǎn)是:它們可以是無限的。這一點(diǎn)和集合類型不一樣,在Java中的集合類型必須是有限的。Stream之所以可以是無限的也是源于Stream「懶」的這一特點(diǎn)。

Stream只會(huì)返回你需要的元素,而不會(huì)一次性地將整個(gè)無限集合返回給你。

Stream接口中有一個(gè)靜態(tài)方法iterate(),這個(gè)方法能夠?yàn)槟銊?chuàng)建一個(gè)無限的Stream對(duì)象。它需要接受兩個(gè)參數(shù):

public static Stream iterate(final T seed, final UnaryOperator f)

其中,seed表示的是這個(gè)無限序列的起點(diǎn),而UnaryOperator則表示的是如何根據(jù)前一個(gè)元素來得到下一個(gè)元素,比如序列中的第二個(gè)元素可以這樣決定:f.apply(seed)。

下面是一個(gè)計(jì)算從某個(gè)數(shù)字開始并依次返回后面count個(gè)素?cái)?shù)的例子: 

  1. public class Primes {      
  2.     public static boolean isPrime(final int number) {  
  3.         return number > 1 &&  
  4.             // 依次從2到number的平方根判斷number是否可以整除該值,即divisor  
  5.             IntStream.rangeClosed(2, (int) Math.sqrt(number))  
  6.                 .noneMatch(divisor -> number % divisor == 0);  
  7.     }   
  8.     private static int primeAfter(final int number) {  
  9.         if(isPrime(number + 1)) // 如果當(dāng)前的數(shù)的下一個(gè)數(shù)是素?cái)?shù),則直接返回該值  
  10.             return number + 1;  
  11.         else // 否則繼續(xù)從下一個(gè)數(shù)據(jù)的后面繼續(xù)找到第一個(gè)素?cái)?shù)返回,遞歸  
  12.             return primeAfter(number + 1);  
  13.     }  
  14.     public static List<Integer> primes(final int fromNumber, final int count) {  
  15.         return Stream.iterate(primeAfter(fromNumber - 1), Primes::primeAfter)  
  16.             .limit(count)  
  17.             .collect(Collectors.<Integer>toList());  
  18.     }  
  19.     //...  

對(duì)于iterate和limit,它們只是中間操作,得到的對(duì)象仍然是Stream類型。對(duì)于collect方法,它是一個(gè)結(jié)束操作,會(huì)觸發(fā)中間操作來得到需要的結(jié)果。

如果用非Stream的方式需要面臨兩個(gè)問題:

  •  一是無法提前知曉fromNumber后count個(gè)素?cái)?shù)的數(shù)值邊界是什么
  •  二是無法使用有限的集合來表示計(jì)算范圍,無法計(jì)算超大的數(shù)值

即不知道第一個(gè)素?cái)?shù)的位置在哪兒,需要提前計(jì)算出來第一個(gè)素?cái)?shù),然后用while來處理count次查找后續(xù)的素?cái)?shù)。可能primes方法的實(shí)現(xiàn)會(huì)拆成兩部分,實(shí)現(xiàn)復(fù)雜。如果用Stream來實(shí)現(xiàn),流式的處理,無限迭代,指定截止條件,內(nèi)部的一套機(jī)制可以保證實(shí)現(xiàn)和執(zhí)行都很優(yōu)雅。 

 

責(zé)任編輯:龐桂玉 來源: Java知音
相關(guān)推薦

2017-05-02 21:08:35

開發(fā)架構(gòu)工程師

2024-12-27 10:51:53

2018-06-29 14:51:41

Java健壯性實(shí)踐

2009-06-18 09:51:25

Java繼承

2022-04-02 14:43:59

Promethues監(jiān)控

2012-09-25 10:03:56

JavaJava封面Java開發(fā)

2011-07-13 09:13:56

Android設(shè)計(jì)

2021-10-21 06:52:17

Vue3組件 API

2011-03-11 09:27:11

Java性能監(jiān)控

2013-04-07 10:40:55

前端框架前端

2011-03-15 17:46:43

2010-11-02 14:11:15

SilverlightWPF微軟開發(fā)

2022-05-19 07:52:39

MLOps云計(jì)算實(shí)機(jī)器學(xué)習(xí)

2022-08-02 09:55:04

React前端

2017-09-05 09:17:47

Java編程用法總結(jié)

2015-12-04 10:04:53

2012-04-19 10:06:55

微軟Windows 8 E

2020-09-28 06:45:42

故障復(fù)盤修復(fù)

2017-12-21 07:54:07

2009-06-04 16:28:43

EJB常見問題
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 国产视频三区 | 精品无码三级在线观看视频 | 国产农村妇女毛片精品久久麻豆 | 日韩在线不卡视频 | aaa在线 | 亚洲欧美日韩精品久久亚洲区 | 韩日在线观看视频 | www国产精| 久久伊人精品一区二区三区 | 91av入口 | 成人午夜电影在线观看 | 成人欧美一区二区三区在线播放 | 亚洲视频一区二区三区 | 日韩高清一区 | 成人欧美一区二区三区色青冈 | 亚洲人成一区二区三区性色 | 久久国产亚洲 | 欧美激情久久久 | 国产精品美女www爽爽爽视频 | 国产精品毛片无码 | 欧美精三区欧美精三区 | 国产精品久久久久一区二区三区 | 九九亚洲 | 国产成人综合在线 | 中文字幕在线第一页 | 亚洲精品视频网站在线观看 | 精品国产一区二区国模嫣然 | 亚洲日本视频 | 一区网站 | 亚洲精品一区二区三区蜜桃久 | 日韩视频在线免费观看 | 一级做a爰片性色毛片 | 精品少妇一区二区三区日产乱码 | 亚洲国产精品成人久久久 | 亚洲人成人一区二区在线观看 | 浴室洗澡偷拍一区二区 | 岛国在线免费观看 | 精品亚洲国产成av人片传媒 | 国产乱码精品1区2区3区 | 久久久一区二区 | 精品福利一区二区三区 |