题目要求

根据需求实现一个缓存池,当请求第一次加载的时候,计算缓存值,并存入缓存中,当另一请求来的时候,直接从缓存中获取对应值,避免重复计算,注意只允许第一次的请求进入计算过程:

实现思路

通过map实现缓存的功能,通过加锁的方式实现只有一个请求能够进入到计算的流程中

  • 缓存工具类
package com.ijianghu.basetype.concurrent;

import java.io.UnsupportedEncodingException;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.locks.ReentrantLock;

/**
 *
 * 设计一个缓存类,当多个线程调用的时候,第一个线程进行计算,并放入缓存;
 * 其他线程直接返回已有的缓存
 *
 */
public class CacheUtils {

    /**
     * 缓存map
     */
    private HashMap<String,String> cacheMap = new HashMap();

    private ReentrantLock lock = new ReentrantLock();


    /**
     * 获取缓存值,只有一个线程能进来
     * @param key
     * @return
     */
    public String getCacheValue(String key){

        if(Objects.isNull(cacheMap.get(key))){
            lock.lock();
            if(Objects.isNull(cacheMap.get(key))){
                String value = calculateValue(key);
                try {
                    Thread.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                cacheMap.put(key,value);
                Set<Map.Entry<String, String>> entries = cacheMap.entrySet();
                for (Map.Entry<String, String> entry: entries) {
                    System.out.println("key:".concat(entry.getKey()).concat("---------value:").concat(entry.getValue()));
                }
                lock.unlock();
                return value;
            }
           else{
                lock.unlock();
                System.out.println(Thread.currentThread().getName().concat(key+"等待得到返回值:".concat(cacheMap.get(key))));
                return cacheMap.get(key);
            }

        }else{
            System.out.println(Thread.currentThread().getName().concat("key+直接得到返回值:".concat(cacheMap.get(key))));
            return cacheMap.get(key);
        }


    }

    public String calculateValue(String key){
        try {

            MessageDigest digest = MessageDigest.getInstance("SHA-1");
            digest.update(key.getBytes("UTF-8"));
            byte[] digestByte = digest.digest();
            System.out.println("digestByte的长度:"+digestByte.length);
            String value = byteToHex(digestByte);
            System.out.println(Thread.currentThread().getName()+key+"计算缓存值:".concat(value));
            return value;
        } catch (NoSuchAlgorithmException | UnsupportedEncodingException e) {
            e.printStackTrace();
        }
        return null;
    }

    private String byteToHex(byte[] digestByte) {
        // 用来将字节转换成 16 进制表示的字符
        char hexDigits[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};
        char str[]  = new char[16*2];
        int k = 0;//表示转换结果中对应的字符位置
        for(int i=0;i<16;i++){
            byte byte0 = digestByte[i];//取第i个字节
            str[k++] = hexDigits[byte0>>> 4 & 0xf];//取字节中高4位的数字转换,>>> 逻辑右移,将符号位一起右移;
            str[k++] = hexDigits[byte0 & 0xf];//取字节中低4位的数字转换
        }

        return new String(str);
    }

}
  • 模拟调用 1
    在调用过程中,为了是主线程等待子线程先执行完,可以有多重实现方法,此类中包含了两种方式:
    1、使用CountDownLatch,使一个线程等待其他线程执行完,再执行;注意使用时,在执行过程中最后在调用countDown()方法
    2、调用线程的join()方法,使主线程进入等待状态(new 、runnable、blocked、waiting、time-waiting、terminated)
package com.ijianghu.basetype.concurrent;

import java.util.Objects;
import java.util.concurrent.CountDownLatch;

/**
 * @Description:create
 * @author: ext.liukai3
 * @date: 2021/6/2 11:08
 *
 * Thread.join()实现主线程阻塞
 *
 * 使用CountDownLatch实现主线程阻塞
 */
public class CacheDemo {

    public static void main(String[] args) {
        CacheUtils cacheUtils = new CacheUtils();
        String[] str = new String[]{"a","b","c","d","e"};
        long start = System.currentTimeMillis();
        CountDownLatch countDownLatch = new CountDownLatch(100);
        System.out.println("start:"+start);
        for(int i=0;i<100;i++){
            Thread thread = new Thread(new CacheThread(str[i%5],cacheUtils,countDownLatch));
            try {


                Thread.sleep(2);
                thread.start();

//                thread.join();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }


        }
        try {
            countDownLatch.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        long end = System.currentTimeMillis();
        System.out.println("end:"+end+"共消耗:"+(end-start));
    }


}
class CacheThread extends Thread{
    private CountDownLatch countDownLatch;
    private  CacheUtils cacheUtils;
    private String key;
    public CacheThread(String key,CacheUtils cacheUtils){
        this.key = key;
        this.cacheUtils = cacheUtils;
    }

    public CacheThread(String key,CacheUtils cacheUtils,CountDownLatch countDownLatch){
        this(key,cacheUtils);
        this.countDownLatch = countDownLatch;
    }
    @Override
    public void run() {
        cacheUtils.getCacheValue(key);
        if(Objects.nonNull(countDownLatch)){
            countDownLatch.countDown();
        }
    }
}
  • 模拟调用2
    通过FutureTask.get()方法阻塞主线程
/**
 */
public class CacheFutureTask implements Callable<String> {

    private CacheUtils cacheUtils;

    private String key;

    public CacheFutureTask(String key,CacheUtils cacheUtils){
        this.cacheUtils = cacheUtils;
        this.key = key;
    }

    @Override
    public String call() throws Exception {
        String cacheValue = cacheUtils.getCacheValue(key);
        return cacheValue;
    }
}

/**
 *
 * FutureTask 使用get()可以实现主线程阻塞
 */
public class CacheFutureDemo {

    public static void main(String[] args) {
        CacheUtils cacheUtils = new CacheUtils();
        String[] str = new String[]{"a","b","c","d","e"};
        long start = System.currentTimeMillis();

        for(int i=0;i<100;i++){
            CacheFutureTask futureTask = new CacheFutureTask(str[i % 5], cacheUtils);
            FutureTask<String> task = new FutureTask<>(futureTask);
            new Thread(task).start();
            String s = null;
            try {
                s = task.get();
            } catch (InterruptedException e) {
                e.printStackTrace();
            } catch (ExecutionException e) {
                e.printStackTrace();
            }
            System.out.println(s);
        }
        long end = System.currentTimeMillis();
        System.out.println("end:"+end+"共消耗:"+(end-start));
    }
}