From 22c359ea29b5ab086369f73ea3ed93529dbdc362 Mon Sep 17 00:00:00 2001
From: zj <1772600164@qq.com>
Date: Tue, 23 Jul 2024 18:43:14 +0800
Subject: [PATCH] 1

---
 websocketSerivce/src/main/java/org/example/WsServerApplication.java             |    8 -
 websocketSerivce/src/main/java/org/example/server/CurrencySerivce.java          |    1 
 websocketSerivce/src/main/java/org/example/pojo/MarketDataOut.java              |   47 +++++
 websocketSerivce/src/main/java/org/example/server/impl/CurrencySerivceImpl.java |  333 +++++++++++++++++++++++------------------
 websocketSerivce/src/main/java/org/example/task/MarketDataTask.java             |   63 +++++++
 websocketSerivce/src/main/java/org/example/websocket/server/WsServer.java       |   21 ++
 6 files changed, 316 insertions(+), 157 deletions(-)

diff --git a/websocketSerivce/src/main/java/org/example/WsServerApplication.java b/websocketSerivce/src/main/java/org/example/WsServerApplication.java
index 76c266b..a2896bd 100644
--- a/websocketSerivce/src/main/java/org/example/WsServerApplication.java
+++ b/websocketSerivce/src/main/java/org/example/WsServerApplication.java
@@ -25,13 +25,5 @@
     public static void main(String[] args) {
         SpringApplication.run(WsServerApplication.class, args);
     }
-
-    @Autowired
-    private CurrencySerivce currencySerivce;
-
-    @Bean
-    public void start() throws JsonProcessingException {
-        currencySerivce.start();
-    }
 }
 
diff --git a/websocketSerivce/src/main/java/org/example/pojo/MarketDataOut.java b/websocketSerivce/src/main/java/org/example/pojo/MarketDataOut.java
new file mode 100644
index 0000000..5ca6371
--- /dev/null
+++ b/websocketSerivce/src/main/java/org/example/pojo/MarketDataOut.java
@@ -0,0 +1,47 @@
+package org.example.pojo;
+
+import lombok.Data;
+
+/**
+ * @program: demo
+ * @description:
+ * @create: 2024-07-23 10:02
+ **/
+@Data
+public class MarketDataOut {
+
+    //币种
+    private String baseAsset;
+
+    //买入平台
+    private String buyingPlatform;
+
+    //卖出平台
+    private String sellPlatform;
+
+    //价差
+    private String spread;
+
+    //买入价格
+    private String buyPrice;
+
+    //卖出价格
+    private String sellPrice;
+
+    //买入数量
+    private String buyNumber;
+
+    //卖出数量
+    private String sellNumber;
+
+    //买入总价
+    private String buyTotalPrice;
+
+    //卖出总价
+    private String sellTotalPrice;
+
+    //服务器时间
+    private String servceTime;
+
+
+}
diff --git a/websocketSerivce/src/main/java/org/example/server/CurrencySerivce.java b/websocketSerivce/src/main/java/org/example/server/CurrencySerivce.java
index 2abdadc..2b2bc41 100644
--- a/websocketSerivce/src/main/java/org/example/server/CurrencySerivce.java
+++ b/websocketSerivce/src/main/java/org/example/server/CurrencySerivce.java
@@ -8,5 +8,4 @@
  * @create: 2024-07-16 15:23
  **/
 public interface CurrencySerivce {
-    void start() throws JsonProcessingException;
 }
diff --git a/websocketSerivce/src/main/java/org/example/server/impl/CurrencySerivceImpl.java b/websocketSerivce/src/main/java/org/example/server/impl/CurrencySerivceImpl.java
index c12b3e8..ecd9266 100644
--- a/websocketSerivce/src/main/java/org/example/server/impl/CurrencySerivceImpl.java
+++ b/websocketSerivce/src/main/java/org/example/server/impl/CurrencySerivceImpl.java
@@ -6,23 +6,35 @@
 import com.fasterxml.jackson.databind.ObjectMapper;
 import com.google.common.reflect.TypeToken;
 import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
 import lombok.Data;
+import lombok.extern.slf4j.Slf4j;
 import org.apache.commons.lang.StringUtils;
 import org.example.dao.CurrencyMapper;
 import org.example.pojo.Currency;
 import org.example.pojo.Market;
+import org.example.pojo.MarketDataOut;
 import org.example.pojo.bo.AsksBo;
 import org.example.pojo.bo.BidsBo;
 import org.example.pojo.bo.MarketBo;
 import org.example.server.CurrencySerivce;
 import org.example.util.RedisUtil;
+import org.example.websocket.server.WsServer;
+import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.stereotype.Service;
 import org.yaml.snakeyaml.error.Mark;
 import java.math.BigDecimal;
 import java.math.RoundingMode;
+import java.time.LocalDateTime;
+import java.time.format.DateTimeFormatter;
 import java.util.*;
 import java.util.ArrayList;
+import java.util.concurrent.*;
+import java.util.concurrent.locks.Lock;
+import java.util.concurrent.locks.ReentrantLock;
 import java.util.stream.Collectors;
+import java.io.ByteArrayOutputStream;
+
 
 /**
  * @program: demo
@@ -30,9 +42,9 @@
  * @create: 2024-07-16 15:23
  **/
 @Service
+@Slf4j
 public class CurrencySerivceImpl extends ServiceImpl<CurrencyMapper, Currency> implements CurrencySerivce {
 
-    private HashMap hashMap = new HashMap();
     private static final Gson gson = new Gson();
     private ArrayList<String> keys;
 
@@ -41,58 +53,72 @@
     List<MarketBo> bitgetList = new ArrayList<>();
     List<MarketBo> kucoinList = new ArrayList<>();
 
-    @Override
-    public void start() throws JsonProcessingException {
+    private final Lock syncCurrencyLock = new ReentrantLock();
+
+    @Autowired
+    private WsServer wsServer;
+
+    private void extracted() {
+        ExecutorService executor = Executors.newFixedThreadPool(4); // 适当增加线程数以匹配处理的交易所数量
+
+        // 异步处理每个交易所的数据
+        CompletableFuture<Void> mexcFuture = CompletableFuture.runAsync(() -> processMexc(), executor);
+        CompletableFuture<Void> gateFuture = CompletableFuture.runAsync(() -> processMarketData(RedisUtil.keys("gate"), gateList, "gate"), executor);
+        CompletableFuture<Void> bitgetFuture = CompletableFuture.runAsync(() -> processMarketData(RedisUtil.keys("bitget"), bitgetList, "bitget"), executor);
+        CompletableFuture<Void> kucoinFuture = CompletableFuture.runAsync(() -> processMarketData(RedisUtil.keys("kucoin"), kucoinList, "kucoin"), executor);
+
+        // 等待所有任务完成
+        CompletableFuture.allOf(mexcFuture, gateFuture, bitgetFuture, kucoinFuture).join();
+
+        executor.shutdown(); // 关闭线程池
+    }
+
+    private void processMexc() {
         Set<String> mexcSet = RedisUtil.keys("mexc");
-        Set<String> gateSet = RedisUtil.keys("gate");
-        Set<String> bitgetSet = RedisUtil.keys("bitget");
-        Set<String> kucoinSet = RedisUtil.keys("kucoin");
 
-
-        //这里做一个定时器,每10分钟更新一次
-        Set<String> setKeys = RedisUtil.keys("*");
-        keys = new ArrayList<>(setKeys);
         for (String key : mexcSet) {
             MarketBo marketBo = new MarketBo();
 
             String v = RedisUtil.get(key);
             Map<String, Object> redisValueMap = gson.fromJson(v, new TypeToken<Map<String, Object>>() {}.getType());
-            String asks = redisValueMap.get("asks").toString();
-            String bids = redisValueMap.get("bids").toString();
+            Object asksObj = redisValueMap.get("asks");
+            Object bidsObj = redisValueMap.get("bids");
 
-            if(!asks.equals("[]") && !StringUtils.isEmpty(asks)){
-                Gson gson = new Gson();
+            if (asksObj instanceof List && !((List<?>) asksObj).isEmpty()) {
+                List<?> asksList = (List<?>) asksObj;
+                Map<?, ?> asksMap = (Map<?, ?>) asksList.get(0);
                 AsksBo asksBo = new AsksBo();
-                Market[] asksDataArray = gson.fromJson(asks, Market[].class);
-                Market asksElement = asksDataArray[0];
-                asksBo.setP(asksElement.getP());
-                asksBo.setV(asksElement.getV());
+                asksBo.setP(new BigDecimal(asksMap.get("p").toString()));
+                asksBo.setV(new BigDecimal(asksMap.get("v").toString()));
                 marketBo.setAsks(asksBo);
             }
 
-            if(!bids.equals("[]") && !StringUtils.isEmpty(bids)){
+            if (bidsObj instanceof List && !((List<?>) bidsObj).isEmpty()) {
+                List<?> bidsList = (List<?>) bidsObj;
+                Map<?, ?> bidsMap = (Map<?, ?>) bidsList.get(bidsList.size() - 1);
                 BidsBo bidsBo = new BidsBo();
-                Market[] bidsDataArray = gson.fromJson(bids, Market[].class);
-                Market bidsElement = bidsDataArray[bidsDataArray.length-1];
-                bidsBo.setP(bidsElement.getP());
-                bidsBo.setV(bidsElement.getV());
+                bidsBo.setP(new BigDecimal(bidsMap.get("p").toString()));
+                bidsBo.setV(new BigDecimal(bidsMap.get("v").toString()));
                 marketBo.setBids(bidsBo);
             }
-            marketBo.setKey(key.replaceAll("mexc",""));
+
+            marketBo.setKey(key.replaceAll("mexc", ""));
             marketBo.setExchange("mexc");
             mexcList.add(marketBo);
         }
+    }
 
-
-        for (String key : gateSet) {
+    private void processMarketData(Set<String> set, List<MarketBo> list, String exchangeName) {
+        for (String key : set) {
             String v = RedisUtil.get(key);
             Map<String, Object> redisValueMap = gson.fromJson(v, new TypeToken<Map<String, Object>>() {}.getType());
-            String asks = redisValueMap.get("asks").toString();
-            String bids = redisValueMap.get("bids").toString();
-
+            Object asksObj = redisValueMap.get("asks");
+            Object bidsObj = redisValueMap.get("bids");
             MarketBo marketBo = new MarketBo();
-            if(!asks.equals("[]") && !StringUtils.isEmpty(asks)){
-                String[][] dataArray = gson.fromJson(asks, String[][].class);
+
+            if (asksObj instanceof List && !((List<?>) asksObj).isEmpty()) {
+                List<?> asksList = (List<?>) asksObj;
+                String[][] dataArray = gson.fromJson(gson.toJson(asksList), String[][].class);
                 String[] asksData = dataArray[0];
                 AsksBo asksBo = new AsksBo();
                 asksBo.setP(new BigDecimal(asksData[0]));
@@ -100,79 +126,20 @@
                 marketBo.setAsks(asksBo);
             }
 
-            if(!bids.equals("[]") && !StringUtils.isEmpty(bids)){
-                String[][] dataArray = gson.fromJson(bids, String[][].class);
-                String[] bidsData = dataArray[dataArray.length-1];
+            if (bidsObj instanceof List && !((List<?>) bidsObj).isEmpty()) {
+                List<?> bidsList = (List<?>) bidsObj;
+                String[][] dataArray = gson.fromJson(gson.toJson(bidsList), String[][].class);
+                String[] bidsData = dataArray[bidsList.size() - 1];
                 BidsBo bidsBo = new BidsBo();
                 bidsBo.setP(new BigDecimal(bidsData[0]));
                 bidsBo.setV(new BigDecimal(bidsData[1]));
                 marketBo.setBids(bidsBo);
             }
-            marketBo.setKey(key.replaceAll("gate",""));
-            marketBo.setExchange("gate");
-            gateList.add(marketBo);
+
+            marketBo.setKey(key.replaceAll(exchangeName, ""));
+            marketBo.setExchange(exchangeName);
+            list.add(marketBo);
         }
-
-
-        for (String key : bitgetSet) {
-            String v = RedisUtil.get(key);
-            Map<String, Object> redisValueMap = gson.fromJson(v, new TypeToken<Map<String, Object>>() {}.getType());
-            String asks = redisValueMap.get("asks").toString();
-            String bids = redisValueMap.get("bids").toString();
-
-            MarketBo marketBo = new MarketBo();
-            if(!asks.equals("[]") && !StringUtils.isEmpty(asks)){
-                String[][] dataArray = gson.fromJson(asks, String[][].class);
-                String[] asksData = dataArray[0];
-                AsksBo asksBo = new AsksBo();
-                asksBo.setP(new BigDecimal(asksData[0]));
-                asksBo.setV(new BigDecimal(asksData[1]));
-                marketBo.setAsks(asksBo);
-            }
-
-            if(!bids.equals("[]") && !StringUtils.isEmpty(bids)){
-                String[][] dataArray = gson.fromJson(bids, String[][].class);
-                String[] bidsData = dataArray[dataArray.length-1];
-                BidsBo bidsBo = new BidsBo();
-                bidsBo.setP(new BigDecimal(bidsData[0]));
-                bidsBo.setV(new BigDecimal(bidsData[1]));
-                marketBo.setBids(bidsBo);
-            }
-            marketBo.setKey(key.replaceAll("bitget",""));
-            marketBo.setExchange("bitget");
-            bitgetList.add(marketBo);
-        }
-
-
-        for (String key : kucoinSet) {
-            String v = RedisUtil.get(key);
-            Map<String, Object> redisValueMap = gson.fromJson(v, new TypeToken<Map<String, Object>>() {}.getType());
-            String asks = redisValueMap.get("asks").toString();
-            String bids = redisValueMap.get("bids").toString();
-
-            MarketBo marketBo = new MarketBo();
-            if(!asks.equals("[]") && !StringUtils.isEmpty(asks)){
-                String[][] dataArray = gson.fromJson(asks, String[][].class);
-                String[] asksData = dataArray[0];
-                AsksBo asksBo = new AsksBo();
-                asksBo.setP(new BigDecimal(asksData[0]));
-                asksBo.setV(new BigDecimal(asksData[1]));
-                marketBo.setAsks(asksBo);
-            }
-
-            if(!bids.equals("[]") && !StringUtils.isEmpty(bids)){
-                String[][] dataArray = gson.fromJson(bids, String[][].class);
-                String[] bidsData = dataArray[dataArray.length-1];
-                BidsBo bidsBo = new BidsBo();
-                bidsBo.setP(new BigDecimal(bidsData[0]));
-                bidsBo.setV(new BigDecimal(bidsData[1]));
-                marketBo.setBids(bidsBo);
-            }
-            marketBo.setKey(key.replaceAll("kucoin",""));
-            marketBo.setExchange("kucoin");
-            kucoinList.add(marketBo);
-        }
-        testDemo();
     }
 
     // 计算利润百分比
@@ -185,74 +152,146 @@
         return profitPercentage;
     }
 
-    private static List<String> findProfitablePairs(List<MarketBo>... exchangeLists) {
-        List<String> result = new ArrayList<>();
+    private void findProfitablePairs(List<MarketBo>... exchangeLists) {
         Map<String, Map<String, MarketBo>> marketMap = new HashMap<>();
+        List<MarketDataOut> marketDataOuts = Collections.synchronizedList(new ArrayList<>()); // 使用线程安全的集合
 
         // 按币种名称和交易所分组市场数据
         for (List<MarketBo> exchangeList : exchangeLists) {
             for (MarketBo market : exchangeList) {
                 String coinName = market.getKey();
-                String exchangeName = market.getExchange(); // 假设有获取交易所名称的方法 getExchange()
-
-                // 如果该币种在该交易所不存在,则创建新的列表
-                if (!marketMap.containsKey(coinName)) {
-                    marketMap.put(coinName, new HashMap<>());
-                }
-                if (!marketMap.get(coinName).containsKey(exchangeName)) {
-                    marketMap.get(coinName).put(exchangeName, new MarketBo());
-                }
-
-                // 将市场数据添加到对应的交易所列表中
-                marketMap.get(coinName).put(exchangeName,market);
+                String exchangeName = market.getExchange();
+                marketMap.computeIfAbsent(coinName, k -> new HashMap<>())
+                        .put(exchangeName, market);
             }
         }
-        marketMap = marketMap.entrySet().stream()
-                .filter(entry -> entry.getValue().size() != 1)
-                .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));
-        // 遍历每个币种
-        for (String coinName : marketMap.keySet()) {
-            Map<String, MarketBo> exchangeMap = marketMap.get(coinName);
-            // 遍历每个交易所
-            for (String exchangeName : exchangeMap.keySet()) {
-                MarketBo markets1 = exchangeMap.get(exchangeName);
-                // 在同一交易所中寻找有差价的交易对
-                for (String exchangeName2 : exchangeMap.keySet()) {
-                    MarketBo markets2 = exchangeMap.get(exchangeName2);
-                    if(markets1.getExchange().equals(markets2.getExchange())){
-                        continue;
-                    }
 
-                    if(null != markets1.getBids() && null != markets2.getAsks()){
-                        BigDecimal buyPrice = markets1.getBids().getP();
-                        BigDecimal sellPrice = markets2.getAsks().getP();
-                        // 计算利润百分比
-                        BigDecimal profitPercentage = calculateProfitPercentage(buyPrice, sellPrice);
+        // 过滤出至少在两个交易所都有的币种
+        marketMap.values().removeIf(exchangeMap -> exchangeMap.size() == 1);
 
-                        // 如果利润大于零,则添加到结果列表
-                        if (profitPercentage.compareTo(BigDecimal.ZERO) > 0) {
-                            // 准备输出字符串
-                            String pair = String.format("%s: 在 %s 以 %s 买入,在 %s 以 %s 卖出,利润为 %.4f%%",
-                                    coinName, markets1.getExchange(), buyPrice, markets2.getExchange(), sellPrice, profitPercentage);
-                            result.add(pair);
+        // 并行计算利润百分比
+        ExecutorService executor = Executors.newFixedThreadPool(Math.min(8, marketMap.size())); // 根据需要调整线程池大小
+
+        List<CompletableFuture<Void>> futures = new ArrayList<>();
+
+        // 获取当前时间,用于后续所有 MarketDataOut 的时间戳
+        LocalDateTime currentTime = LocalDateTime.now();
+        String formattedDateTime = currentTime.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")); // 格式化时间字符串
+
+        for (Map.Entry<String, Map<String, MarketBo>> entry : marketMap.entrySet()) {
+            String coinName = entry.getKey();
+            Map<String, MarketBo> exchangeMap = entry.getValue();
+            String[] exchanges = exchangeMap.keySet().toArray(new String[0]);
+
+            for (int i = 0; i < exchanges.length; i++) {
+                MarketBo markets1 = exchangeMap.get(exchanges[i]);
+                if (markets1.getBids() == null) continue;
+
+                for (int j = 0; j < exchanges.length; j++) {
+                    if (i == j) continue;
+
+                    MarketBo markets2 = exchangeMap.get(exchanges[j]);
+                    if (markets2.getAsks() == null) continue;
+
+                    CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
+                        try {
+                            BigDecimal buyPrice = markets1.getBids().getP();
+                            BigDecimal sellPrice = markets2.getAsks().getP();
+                            BigDecimal profitPercentage = calculateProfitPercentage(buyPrice, sellPrice);
+
+                            // 只有在利润百分比大于0时才组装市场数据
+                            if (profitPercentage.compareTo(BigDecimal.ZERO) > 0) {
+                                assembleMarketDataOut(coinName, markets1, markets2, profitPercentage, buyPrice, sellPrice, marketDataOuts, formattedDateTime);
+                            }
+                        } catch (Exception e) {
+                            e.printStackTrace(); // 输出异常信息
                         }
-                    }
+                    }, executor);
+
+                    futures.add(future); // 收集所有 Future
                 }
             }
         }
-        return result;
+
+        // 等待所有任务完成
+        CompletableFuture<Void> allFutures = CompletableFuture.allOf(futures.toArray(new CompletableFuture[0]));
+        allFutures.join();
+
+        executor.shutdown(); // 关闭线程池
+
+        pushWs(marketDataOuts); // 推送结果
     }
 
+    private static void assembleMarketDataOut(String coinName, MarketBo markets1, MarketBo markets2, BigDecimal profitPercentage, BigDecimal buyPrice, BigDecimal sellPrice, List<MarketDataOut> marketDataOuts, String formattedDateTime) {
+        MarketDataOut marketDataOut = new MarketDataOut();
+        marketDataOut.setBaseAsset(coinName.replaceAll("USDT","").toLowerCase()); // 设置基础资产
+        marketDataOut.setBuyingPlatform(markets1.getExchange()); // 设置买入平台
+        marketDataOut.setSellPlatform(markets2.getExchange()); // 设置卖出平台
+        marketDataOut.setSpread(profitPercentage.toString()); // 设置利润百分比
+        marketDataOut.setBuyPrice(buyPrice.toString()); // 设置买入价格
+        marketDataOut.setSellPrice(sellPrice.toString()); // 设置卖出价格
+        marketDataOut.setBuyNumber(markets1.getBids().getV().toString()); // 设置买入数量
+        marketDataOut.setSellNumber(markets2.getAsks().getV().toString()); // 设置卖出数量
+        marketDataOut.setBuyTotalPrice(markets1.getBids().getP().multiply(markets1.getBids().getV()).toString()); // 设置买入总价
+        marketDataOut.setSellTotalPrice(markets2.getAsks().getP().multiply(markets2.getAsks().getV()).toString()); // 设置卖出总价
+        marketDataOut.setServceTime(formattedDateTime); // 设置服务时间
+        marketDataOuts.add(marketDataOut); // 添加到输出列表
+    }
 
-
-    public void testDemo(){
-        List<String> profitablePairs = findProfitablePairs(mexcList, gateList, bitgetList, kucoinList);
-
-        // 输出结果
-        for (String pair : profitablePairs) {
-            System.out.println(pair);
+    private void pushWs(List<MarketDataOut> marketDataOuts) {
+//        String key = "MARKET_Date";
+//        // 创建 Gson 对象
+        Gson gson = new GsonBuilder().setPrettyPrinting().create();
+        // 将列表转换为 JSON 字符串
+        String json = gson.toJson(marketDataOuts);
+//        RedisUtil.set(key,json);
+        try {
+            // 全体发送
+            wsServer.sendMessageToAll(json);
+        } catch (Exception e) {
+            e.printStackTrace();
         }
     }
 
+    public void quotationCalculation(){
+        extracted();
+        findProfitablePairs(mexcList, gateList, bitgetList, kucoinList);
+    }
 
+    public void scheduler(){
+        ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
+
+        // 获取当前时间
+        Calendar now = Calendar.getInstance();
+        int currentHour = now.get(Calendar.HOUR_OF_DAY);
+        int currentMinute = now.get(Calendar.MINUTE);
+
+        // 计算距离下一个1点的小时数和分钟数
+        int hourToSchedule = 1;
+        int minuteToSchedule = 0;
+        if (currentHour > hourToSchedule || (currentHour == hourToSchedule && currentMinute >= minuteToSchedule)) {
+            now.add(Calendar.DAY_OF_MONTH, 1); // 如果当前时间已过1点,则将调度时间设为明天的1点
+        }
+        now.set(Calendar.HOUR_OF_DAY, hourToSchedule);
+        now.set(Calendar.MINUTE, minuteToSchedule);
+        now.set(Calendar.SECOND, 0);
+        now.set(Calendar.MILLISECOND, 0);
+
+        Date scheduledTime = now.getTime();
+        long initialDelay = scheduledTime.getTime() - System.currentTimeMillis();
+
+        // 调度任务
+        scheduler.scheduleAtFixedRate(task, initialDelay, TimeUnit.DAYS.toMillis(1), TimeUnit.MILLISECONDS);
+    }
+
+    Runnable task = new Runnable() {
+        @Override
+        public void run() {
+
+            // 获取所有 Redis 键
+            Set<String> setKeys = RedisUtil.keys("*");
+            keys = new ArrayList<>(setKeys);
+            log.info("---------------------更新交易对完成---------------------");
+        }
+    };
 }
diff --git a/websocketSerivce/src/main/java/org/example/task/MarketDataTask.java b/websocketSerivce/src/main/java/org/example/task/MarketDataTask.java
new file mode 100644
index 0000000..a0dfba6
--- /dev/null
+++ b/websocketSerivce/src/main/java/org/example/task/MarketDataTask.java
@@ -0,0 +1,63 @@
+package org.example.task;
+
+import lombok.extern.slf4j.Slf4j;
+import org.example.server.impl.CurrencySerivceImpl;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.scheduling.annotation.Scheduled;
+import org.springframework.stereotype.Component;
+
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.Executors;
+import java.util.concurrent.locks.Lock;
+import java.util.concurrent.locks.ReentrantLock;
+
+/**
+ * @program: demo
+ * @description:
+ * @create: 2024-07-23 18:01
+ **/
+@Component
+@Slf4j
+public class MarketDataTask {
+
+    private final ExecutorService executor = Executors.newSingleThreadExecutor(); // 创建单线程线程池
+
+    @Autowired
+    private CurrencySerivceImpl currencySerivce;
+
+    private final Lock syncCurrencyLock = new ReentrantLock();
+
+    @Scheduled(initialDelay = 0, fixedRate = Long.MAX_VALUE)
+    public void start() {
+        log.info("行情数据计算-------------启动");
+
+        // 启动任务调度器
+//        currencySerivce.scheduler();
+
+        executor.execute(() -> {
+            try {
+                while (true) {
+                    syncCurrencyLock.lock();
+                    try {
+                        long startTime = System.nanoTime();
+                        currencySerivce.quotationCalculation();
+                        long endTime = System.nanoTime();
+                        double durationInSeconds = (endTime - startTime) / 1_000_000_000.0;
+                        if(durationInSeconds >= 2){
+                            System.out.println("程序执行时间: " + durationInSeconds + " 秒");
+                        }
+                    } catch (Exception e) {
+                        log.error("行情数据计算错误--------------【严重预警】");
+                        e.printStackTrace();
+                    } finally {
+                        syncCurrencyLock.unlock();
+                    }
+                }
+            } catch (Exception e) {
+                log.error("行情数据计算--------------线程被中断");
+                Thread.currentThread().interrupt();
+            }
+        });
+    }
+
+}
diff --git a/websocketSerivce/src/main/java/org/example/websocket/server/WsServer.java b/websocketSerivce/src/main/java/org/example/websocket/server/WsServer.java
index 9ae676e..73af220 100644
--- a/websocketSerivce/src/main/java/org/example/websocket/server/WsServer.java
+++ b/websocketSerivce/src/main/java/org/example/websocket/server/WsServer.java
@@ -10,17 +10,21 @@
 import org.springframework.stereotype.Component;
 import org.springframework.web.bind.annotation.PostMapping;
 import org.springframework.web.bind.annotation.RequestBody;
+import java.io.ByteArrayOutputStream;
 
 import javax.websocket.*;
 import javax.websocket.server.ServerEndpoint;
 import java.io.IOException;
 import java.lang.reflect.Type;
+import java.nio.ByteBuffer;
 import java.text.SimpleDateFormat;
 import java.util.*;
 import java.util.concurrent.*;
 import java.util.concurrent.atomic.AtomicInteger;
 import java.util.concurrent.locks.Lock;
 import java.util.concurrent.locks.ReentrantLock;
+import java.util.zip.Deflater;
+import java.util.zip.DeflaterOutputStream;
 
 /**
  * @ClassDescription: websocket服务端
@@ -82,7 +86,12 @@
                         sessionLock.lock();
                         try {
                             synchronized (session){
-                                session.getBasicRemote().sendText(message);
+                                // 压缩消息
+                                byte[] compressedData = compress(message);
+
+                                // 发送压缩后的消息
+                                session.getBasicRemote().sendBinary(ByteBuffer.wrap(compressedData));
+//                                session.getBasicRemote().sendText(message);
                             }
                         } catch (Exception e) {
                             log.error("发送消息时出现异常: " + e.getMessage());
@@ -109,6 +118,16 @@
         }
     }
 
+    private byte[] compress(String data) throws IOException {
+        ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
+        Deflater deflater = new Deflater(Deflater.BEST_COMPRESSION, true);
+        try (DeflaterOutputStream deflaterOutputStream = new DeflaterOutputStream(byteArrayOutputStream, deflater)) {
+            deflaterOutputStream.write(data.getBytes("UTF-8"));
+        }
+        deflater.end();
+        return byteArrayOutputStream.toByteArray();
+    }
+
 //
 //    public static Map<String, Object> jsonToMap(String json) {
 //        Gson gson = new Gson();

--
Gitblit v1.9.3