guava 的 V get(K key, Callable<? extends V> loader) 在执行耗时的缓存加载期间会返回之前的缓存吗?

暗中观察 发布于 2019/05/26 10:03
阅读 1K+
收藏 0

开源之夏第三届火热来袭,高校学生参与赢万元奖金!>>>

代码


import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import org.apache.commons.lang3.time.FastDateFormat;

import java.text.ParseException;
import java.util.Date;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicInteger;

public class JVMTest {


    static Cache<String,Integer> cache=CacheBuilder.newBuilder()
            .expireAfterWrite(200,TimeUnit.MILLISECONDS)
            .build();

    public static void main(String[] args) throws  ParseException {
        String key="a";

        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                long start = System.currentTimeMillis();
                while (true){
                    Integer cacheVal = loaderCache(key);
                    System.out.println("thread:"+Thread.currentThread()+",val:"+cacheVal);
                    long end = System.currentTimeMillis();
                    if(end - start > 3000){
                        break;
                    }
                }
            }
        };
        try {
            ExecutorService pool = Executors.newFixedThreadPool(10);
            for (int i = 0; i < 10; i++) {
                pool.execute(runnable);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

    }


    static Integer loaderCache(String key){
        Integer cacheVal = null;
        try {
            cacheVal = cache.get(key,()->loader(key));
//            cacheVal = cache.get(key);
            System.out.println("cache-thread:"+Thread.currentThread()+"cacheVal:"+cacheVal);
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
      return  cacheVal;
    }

    static Integer loader(String key){
        System.out.println("threa:"+Thread.currentThread()+",开始耗时加载");
        try {
            TimeUnit.SECONDS.sleep(1);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        int i = ThreadLocalRandom.current().nextInt();
        System.out.println("threa:"+Thread.currentThread()+",加载到的缓存值:"+i);
        return i;
    }
}

控制台信息如下,

从控制台的消息来看,在执行耗时的缓存加载时,其他线程来加载缓存时,会返回旧的缓存值,如果是这样,那真是挺好的

加载中
0
暗中观察
暗中观察

这样就不会出现惊群现象了

0
kakai
kakai

这个很好理解的,cache.get方法,第一个参数是key,第二个参数是一个回调函数,当通过第一个参数获取value时,如果存在则直接返回旧值,然后执行回调函数写入新值。

OSCHINA
登录后可查看更多优质内容
返回顶部
顶部
返回顶部
顶部