1
zyy
2026-03-13 677997551f0de1a8efd754fab1485c02fe560e94
src/main/java/com/nq/service/impl/StockServiceImpl.java
@@ -40,6 +40,8 @@
import java.text.SimpleDateFormat;
import java.util.*;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.locks.ReentrantLock;
import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@@ -177,23 +179,6 @@
        if (stockList.size() > 0){
            stockListVOS.addAll(Objects.requireNonNull(StockApi.getStockReailTimes(stockList)));
        }
        // Create a list to hold Futures
        List<CompletableFuture<Void>> futures = new ArrayList<>();
        // Use CompletableFuture to perform asynchronous processing for each stockListVO
//        for (StockListVO stockListVO : stockListVOS) {
//            CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
//                stockListVO.setNowPrice(iPriceServices.getNowPrice(stockListVO.getCode(), stockType).toString());
//                Map<String, Object> newStock = iPriceServices.getNewStock(stockListVO.getCode());
//                stockListVO.setHcrateP(newStock.get("pcp").toString());
//            });
//            futures.add(future);
//        }
//
//        // Wait for all futures to complete
//        CompletableFuture.allOf(futures.toArray(new CompletableFuture[0])).join();
        RPageInfo pageInfo = new RPageInfo();
        pageInfo.setList(stockListVOS);
        pageInfo.setStockType(stockType);
@@ -223,9 +208,9 @@
        stockVO.setSpell(stock.getStockSpell());
        stockVO.setGid(stock.getStockGid().toUpperCase());
        Map<String, Object> newStock = iPriceServices.getNewStock(stock.getStockCode());
        stockVO.setHcrate(newStock.get("pcp").toString().replaceAll("%", ""));
        stockVO.setToday_max(newStock.get("high").toString());
        stockVO.setToday_min(newStock.get("low").toString());
        stockVO.setHcrate(newStock.get("ChgPct").toString().replaceAll("%", ""));
        stockVO.setToday_max(newStock.get("High").toString());
        stockVO.setToday_min(newStock.get("Low").toString());
        if(null != cacheBaseStock){
            stockVO.setOpen_px(cacheBaseStock.getOpen());
            stockVO.setPreclose_px(cacheBaseStock.getPrevClose());
@@ -449,29 +434,146 @@
    }
        /*股票日线-K线*/
    // 缓存键类,包含三个参数
    private static class CacheKey {
        private final String pid;
        private final String interval;
        private final String stockType;
        public CacheKey(String pid, String interval, String stockType) {
            this.pid = pid;
            this.interval = interval;
            this.stockType = stockType;
        }
        @Override
        public boolean equals(Object o) {
            if (this == o) return true;
            if (o == null || getClass() != o.getClass()) return false;
            CacheKey cacheKey = (CacheKey) o;
            return pid.equals(cacheKey.pid) &&
                    interval.equals(cacheKey.interval) &&
                    stockType.equals(cacheKey.stockType);
        }
        @Override
        public int hashCode() {
            int result = pid.hashCode();
            result = 31 * result + interval.hashCode();
            result = 31 * result + stockType.hashCode();
            return result;
        }
        @Override
        public String toString() {
            return "CacheKey{pid='" + pid + "', interval='" + interval + "', stockType='" + stockType + "'}";
        }
    }
    // 缓存条目类
    private static class CacheEntry {
        Object result;
        long lastExecuteTime;
        CacheEntry(Object result, long lastExecuteTime) {
            this.result = result;
            this.lastExecuteTime = lastExecuteTime;
        }
    }
    // 缓存存储:每个CacheKey对应一个CacheEntry
    private static final Map<CacheKey, CacheEntry> cacheMap = new ConcurrentHashMap<>();
    // 锁对象存储:每个CacheKey有自己的锁
    private static final Map<CacheKey, Object> lockMap = new ConcurrentHashMap<>();
    private static final long MIN_INTERVAL_MS = 3000;
    @Override
    public Object getKData(String pid, String interval, String stockType) {
        EStockType eStockType = EStockType.getEStockTypeByCode(stockType);
        if(eStockType == EStockType.IN){
            Object object = HttpUtil.get(eStockType.stockUrl + "api/all/getKData.do?pid=" + pid + "&interval=" + interval + "&stockType=in");
            Gson gson = new Gson();
            List<kData> dataList = gson.fromJson(object.toString(), new TypeToken<List<kData>>(){}.getType());
        CacheKey cacheKey = new CacheKey(pid, interval, stockType);
        long currentTime = System.currentTimeMillis();
            Stock stock = stockMapper.selectOne(new LambdaQueryWrapper<Stock>().eq(Stock::getStockCode, pid).eq(Stock::getStockType, "IN"));
            BigDecimal nowPrice = iPriceServices.getNowPrice(stock.getStockCode());
            Map singleStock = getSingleStock(stock.getStockCode());
            StockVO stockVO = (StockVO)singleStock.get("stock");
            // 修改 List 中的最后一条数据
            kData lastData = dataList.get(dataList.size() - 1);
            lastData.setC(nowPrice.toString());
            lastData.setO(stockVO.getOpen_px());
            lastData.setH(stockVO.getToday_max());
            lastData.setL(stockVO.getToday_min());
        // 1. 快速检查缓存(无锁)
        CacheEntry cached = cacheMap.get(cacheKey);
        if (cached != null && (currentTime - cached.lastExecuteTime) < MIN_INTERVAL_MS) {
            return cached.result;
        }
        // 2. 获取该缓存键对应的锁
        Object keyLock = lockMap.computeIfAbsent(cacheKey, k -> new Object());
        // 3. 同步块内再次检查
        synchronized (keyLock) {
            currentTime = System.currentTimeMillis();
            cached = cacheMap.get(cacheKey);
            if (cached != null && (currentTime - cached.lastExecuteTime) < MIN_INTERVAL_MS) {
                return cached.result;
            }
            // 4. 获取新数据
            Object newResult = doGetKData(pid, interval, stockType);
            // 5. 更新缓存
            if (newResult != null) {
                cacheMap.put(cacheKey, new CacheEntry(newResult, System.currentTimeMillis()));
                return newResult;
            } else if (cached != null) {
                // 请求失败,返回旧缓存
                return cached.result;
            }
            return "{\"error\":\"获取K线数据失败\"}";
        }
    }
    private Object doGetKData(String pid, String interval, String stockType) {
        try {
            EStockType eStockType = EStockType.getEStockTypeByCode(stockType);
            String url = eStockType.stockUrl + "kline?pid=" + pid +
                    "&interval=" + interval + "&key=" + eStockType.stockKey;
            Object object = HttpUtil.get(url);
            Gson gson = new Gson();
            List<kData> dataList = gson.fromJson(object.toString(),
                    new TypeToken<List<kData>>(){}.getType());
            // 补充实时数据
            if (dataList != null && !dataList.isEmpty()) {
                enrichWithRealTimeData(pid, dataList);
            }
            return gson.toJson(dataList);
        } catch (Exception e) {
            e.printStackTrace();
            return null; // 返回null表示失败
        }
        return  HttpUtil.get(eStockType.stockUrl + "kline?pid=" + pid + "&interval=" + interval + "&key=" + eStockType.stockKey);
    }
    private void enrichWithRealTimeData(String pid, List<kData> dataList) {
        try {
            Stock stock = stockMapper.selectOne(new LambdaQueryWrapper<Stock>()
                    .eq(Stock::getStockCode, pid)
                    .eq(Stock::getStockType, "IN"));
            if (stock != null) {
                BigDecimal nowPrice = iPriceServices.getNowPrice(stock.getStockCode());
                Map singleStock = getSingleStock(stock.getStockCode());
                StockVO stockVO = (StockVO)singleStock.get("stock");
                kData lastData = dataList.get(dataList.size() - 1);
                lastData.setC(nowPrice.toString());
                lastData.setO(stockVO.getOpen_px());
                lastData.setH(stockVO.getToday_max());
                lastData.setL(stockVO.getToday_min());
            }
        } catch (Exception e) {
            // 实时数据获取失败不影响K线数据返回
            e.printStackTrace();
        }
    }
    @Override