当前位置:   article > 正文

面试官问:说一说高并发场景下的接口请求合并方案?

resttemplate高并发

前言

请求合并到底有什么意义呢?我们来看下图。

b58acb1ac84247a629dd0dc34a0d0a65.png

假设我们3个用户(用户id分别是1、2、3),现在他们都要查询自己的基本信息,请求到服务器,服务器端请求数据库,发出3次请求。我们都知道数据库连接资源是相当宝贵的,那么我们怎么尽可能节省连接资源呢?

这里把数据库换成被调用的远程服务,也是同样的道理。

我们改变下思路,如下图所示。

3d602c0f3bd4639ef8ba29ccc0c170fd.png

我们在服务器端把请求合并,只发出一条SQL查询数据库,数据库返回后,服务器端处理返回数据,根据一个唯一请求ID,把数据分组,返回给对应用户。

技术手段

  • LinkedBlockQueue 阻塞队列

  • ScheduledThreadPoolExecutor 定时任务线程池

  • CompleteableFuture future 阻塞机制(Java 8 的 CompletableFuture 并没有 timeout 机制,后面优化,使用了队列替代)

代码实现

查询用户的代码
  1. public interface UserService {
  2.     Map<String, Users> queryUserByIdBatch(List<UserWrapBatchService.Request> userReqs);
  3. }
  1. @Service
  2. public class UserServiceImpl implements UserService {
  3.     @Resource
  4.     private UsersMapper usersMapper;
  5.     @Override
  6.     public Map<String, Users> queryUserByIdBatch(List<UserWrapBatchService.Request> userReqs) {
  7.         // 全部参数
  8.         List<Long> userIds = userReqs.stream().map(UserWrapBatchService.Request::getUserId).collect(Collectors.toList());
  9.         QueryWrapper<Users> queryWrapper = new QueryWrapper<>();
  10.         // 用in语句合并成一条SQL,避免多次请求数据库的IO
  11.         queryWrapper.in("id", userIds);
  12.         List<Users> users = usersMapper.selectList(queryWrapper);
  13.         Map<Long, List<Users>> userGroup = users.stream().collect(Collectors.groupingBy(Users::getId));
  14.         HashMap<String, Users> result = new HashMap<>();
  15.         userReqs.forEach(val -> {
  16.             List<Users> usersList = userGroup.get(val.getUserId());
  17.             if (!CollectionUtils.isEmpty(usersList)) {
  18.                 result.put(val.getRequestId(), usersList.get(0));
  19.             } else {
  20.                 // 表示没数据
  21.                 result.put(val.getRequestId(), null);
  22.             }
  23.         });
  24.         return result;
  25.     }
  26. }
合并请求的实现,更多面试资料,公众 号Java精选,回复java面试,获取面试资料,支持在线刷题。
  1. package com.springboot.sample.service.impl;
  2. import com.springboot.sample.bean.Users;
  3. import com.springboot.sample.service.UserService;
  4. import org.springframework.stereotype.Service;
  5. import javax.annotation.PostConstruct;
  6. import javax.annotation.Resource;
  7. import java.util.*;
  8. import java.util.concurrent.*;
  9. /***
  10.  * zzq
  11.  * 包装成批量执行的地方
  12.  * */
  13. @Service
  14. public class UserWrapBatchService {
  15.     @Resource
  16.     private UserService userService;
  17.     /**
  18.      * 最大任务数
  19.      **/
  20.     public static int MAX_TASK_NUM = 100;
  21.     /**
  22.      * 请求类,code为查询的共同特征,例如查询商品,通过不同id的来区分
  23.      * CompletableFuture将处理结果返回
  24.      */
  25.     public class Request {
  26.         // 请求id 唯一
  27.         String requestId;
  28.         // 参数
  29.         Long userId;
  30.         //TODO Java 8 的 CompletableFuture 并没有 timeout 机制
  31.         CompletableFuture<Users> completableFuture;
  32.         public String getRequestId() {
  33.             return requestId;
  34.         }
  35.         public void setRequestId(String requestId) {
  36.             this.requestId = requestId;
  37.         }
  38.         public Long getUserId() {
  39.             return userId;
  40.         }
  41.         public void setUserId(Long userId) {
  42.             this.userId = userId;
  43.         }
  44.         public CompletableFuture getCompletableFuture() {
  45.             return completableFuture;
  46.         }
  47.         public void setCompletableFuture(CompletableFuture completableFuture) {
  48.             this.completableFuture = completableFuture;
  49.         }
  50.     }
  51.     /*
  52.     LinkedBlockingQueue是一个阻塞的队列,内部采用链表的结果,通过两个ReenTrantLock来保证线程安全
  53.     LinkedBlockingQueue与ArrayBlockingQueue的区别
  54.     ArrayBlockingQueue默认指定了长度,而LinkedBlockingQueue的默认长度是Integer.MAX_VALUE,也就是无界队列,在移除的速度小于添加的速度时,容易造成OOM。
  55.     ArrayBlockingQueue的存储容器是数组,而LinkedBlockingQueue是存储容器是链表
  56.     两者的实现队列添加或移除的锁不一样,ArrayBlockingQueue实现的队列中的锁是没有分离的,即添加操作和移除操作采用的同一个ReenterLock锁,
  57.     而LinkedBlockingQueue实现的队列中的锁是分离的,其添加采用的是putLock,移除采用的则是takeLock,这样能大大提高队列的吞吐量,
  58.     也意味着在高并发的情况下生产者和消费者可以并行地操作队列中的数据,以此来提高整个队列的并发性能。
  59.      */
  60.     private final Queue<Request> queue = new LinkedBlockingQueue();
  61.     @PostConstruct
  62.     public void init() {
  63.         //定时任务线程池,创建一个支持定时、周期性或延时任务的限定线程数目(这里传入的是1)的线程池
  64.         ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
  65.         scheduledExecutorService.scheduleAtFixedRate(() -> {
  66.             int size = queue.size();
  67.             //如果队列没数据,表示这段时间没有请求,直接返回
  68.             if (size == 0) {
  69.                 return;
  70.             }
  71.             List<Request> list = new ArrayList<>();
  72.             System.out.println("合并了 [" + size + "] 个请求");
  73.             //将队列的请求消费到一个集合保存
  74.             for (int i = 0; i < size; i++) {
  75.                 // 后面的SQL语句是有长度限制的,所以还要做限制每次批量的数量,超过最大任务数,等下次执行
  76.                 if (i < MAX_TASK_NUM) {
  77.                     list.add(queue.poll());
  78.                 }
  79.             }
  80.             //拿到我们需要去数据库查询的特征,保存为集合
  81.             List<Request> userReqs = new ArrayList<>();
  82.             for (Request request : list) {
  83.                 userReqs.add(request);
  84.             }
  85.             //将参数传入service处理, 这里是本地服务,也可以把userService 看成RPC之类的远程调用
  86.             Map<String, Users> response = userService.queryUserByIdBatch(userReqs);
  87.             //将处理结果返回各自的请求
  88.             for (Request request : list) {
  89.                 Users result = response.get(request.requestId);
  90.                 request.completableFuture.complete(result);    //completableFuture.complete方法完成赋值,这一步执行完毕,下面future.get()阻塞的请求可以继续执行了
  91.             }
  92.         }, 10010, TimeUnit.MILLISECONDS);
  93.         //scheduleAtFixedRate是周期性执行 schedule是延迟执行 initialDelay是初始延迟 period是周期间隔 后面是单位
  94.         //这里我写的是 初始化后100毫秒后执行,周期性执行10毫秒执行一次
  95.     }
  96.     public Users queryUser(Long userId) {
  97.         Request request = new Request();
  98.         // 这里用UUID做请求id
  99.         request.requestId = UUID.randomUUID().toString().replace("-""");
  100.         request.userId = userId;
  101.         CompletableFuture<Users> future = new CompletableFuture<>();
  102.         request.completableFuture = future;
  103.         //将对象传入队列
  104.         queue.offer(request);
  105.         //如果这时候没完成赋值,那么就会阻塞,直到能够拿到值
  106.         try {
  107.             return future.get();
  108.         } catch (InterruptedException e) {
  109.             e.printStackTrace();
  110.         } catch (ExecutionException e) {
  111.             e.printStackTrace();
  112.         }
  113.         return null;
  114.     }
  115. }
控制层调用
  1. /***
  2.  * 请求合并
  3.  * */
  4. @RequestMapping("/merge")
  5. public Callable<Users> merge(Long userId) {
  6.     return new Callable<Users>() {
  7.         @Override
  8.         public Users call() throws Exception {
  9.             return userBatchService.queryUser(userId);
  10.         }
  11.     };
  12. }

Callable是什么可以参考:

  • https://blog.csdn.net/baidu_19473529/article/details/123596792

模拟高并发查询的代码
  1. package com.springboot.sample;
  2. import org.springframework.web.client.RestTemplate;
  3. import java.util.Random;
  4. import java.util.concurrent.CountDownLatch;
  5. public class TestBatch {
  6.     private static int threadCount = 30;
  7.     private final static CountDownLatch COUNT_DOWN_LATCH = new CountDownLatch(threadCount); //为保证30个线程同时并发运行
  8.     private static final RestTemplate restTemplate = new RestTemplate();
  9.     public static void main(String[] args) {
  10.         for (int i = 0; i < threadCount; i++) {//循环开30个线程
  11.             new Thread(new Runnable() {
  12.                 public void run() {
  13.                     COUNT_DOWN_LATCH.countDown();//每次减一
  14.                     try {
  15.                         COUNT_DOWN_LATCH.await(); //此处等待状态,为了让30个线程同时进行
  16.                     } catch (InterruptedException e) {
  17.                         e.printStackTrace();
  18.                     }
  19.                     for (int j = 1; j <= 3; j++) {
  20.                         int param = new Random().nextInt(4);
  21.                         if (param <=0){
  22.                             param++;
  23.                         }
  24.                         String responseBody = restTemplate.getForObject("http://localhost:8080/asyncAndMerge/merge?userId=" + param, String.class);
  25.                         System.out.println(Thread.currentThread().getName() + "参数 " + param + " 返回值 " + responseBody);
  26.                     }
  27.                 }
  28.             }).start();
  29.         }
  30.     }
  31. }
测试效果
025f6c92e12c4903223c2be16c7c4b31.png 8d8595957ddbc75b21e882269e2f473d.png
要注意的问题
  • Java 8 的 CompletableFuture 并没有 timeout 机制

  • 后面的SQL语句是有长度限制的,所以还要做限制每次批量的数量,超过最大任务数,等下次执行(本例中加了MAX_TASK_NUM判断)

使用队列的超时解决Java 8 的 CompletableFuture 并没有 timeout 机制

核心代码
  1. package com.springboot.sample.service.impl;
  2. import com.springboot.sample.bean.Users;
  3. import com.springboot.sample.service.UserService;
  4. import org.springframework.stereotype.Service;
  5. import javax.annotation.PostConstruct;
  6. import javax.annotation.Resource;
  7. import java.util.*;
  8. import java.util.concurrent.*;
  9. /***
  10.  * zzq
  11.  * 包装成批量执行的地方,使用queue解决超时问题
  12.  * */
  13. @Service
  14. public class UserWrapBatchQueueService {
  15.     @Resource
  16.     private UserService userService;
  17.     /**
  18.      * 最大任务数
  19.      **/
  20.     public static int MAX_TASK_NUM = 100;
  21.     /**
  22.      * 请求类,code为查询的共同特征,例如查询商品,通过不同id的来区分
  23.      * CompletableFuture将处理结果返回
  24.      */
  25.     public class Request {
  26.         // 请求id
  27.         String requestId;
  28.         // 参数
  29.         Long userId;
  30.         // 队列,这个有超时机制
  31.         LinkedBlockingQueue<Users> usersQueue;
  32.         public String getRequestId() {
  33.             return requestId;
  34.         }
  35.         public void setRequestId(String requestId) {
  36.             this.requestId = requestId;
  37.         }
  38.         public Long getUserId() {
  39.             return userId;
  40.         }
  41.         public void setUserId(Long userId) {
  42.             this.userId = userId;
  43.         }
  44.         public LinkedBlockingQueue<Users> getUsersQueue() {
  45.             return usersQueue;
  46.         }
  47.         public void setUsersQueue(LinkedBlockingQueue<Users> usersQueue) {
  48.             this.usersQueue = usersQueue;
  49.         }
  50.     }
  51.     /*
  52.     LinkedBlockingQueue是一个阻塞的队列,内部采用链表的结果,通过两个ReenTrantLock来保证线程安全
  53.     LinkedBlockingQueue与ArrayBlockingQueue的区别
  54.     ArrayBlockingQueue默认指定了长度,而LinkedBlockingQueue的默认长度是Integer.MAX_VALUE,也就是无界队列,在移除的速度小于添加的速度时,容易造成OOM。
  55.     ArrayBlockingQueue的存储容器是数组,而LinkedBlockingQueue是存储容器是链表
  56.     两者的实现队列添加或移除的锁不一样,ArrayBlockingQueue实现的队列中的锁是没有分离的,即添加操作和移除操作采用的同一个ReenterLock锁,
  57.     而LinkedBlockingQueue实现的队列中的锁是分离的,其添加采用的是putLock,移除采用的则是takeLock,这样能大大提高队列的吞吐量,
  58.     也意味着在高并发的情况下生产者和消费者可以并行地操作队列中的数据,以此来提高整个队列的并发性能。
  59.      */
  60.     private final Queue<Request> queue = new LinkedBlockingQueue();
  61.     @PostConstruct
  62.     public void init() {
  63.         //定时任务线程池,创建一个支持定时、周期性或延时任务的限定线程数目(这里传入的是1)的线程池
  64.         ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);
  65.         scheduledExecutorService.scheduleAtFixedRate(() -> {
  66.             int size = queue.size();
  67.             //如果队列没数据,表示这段时间没有请求,直接返回
  68.             if (size == 0) {
  69.                 return;
  70.             }
  71.             List<Request> list = new ArrayList<>();
  72.             System.out.println("合并了 [" + size + "] 个请求");
  73.             //将队列的请求消费到一个集合保存
  74.             for (int i = 0; i < size; i++) {
  75.                 // 后面的SQL语句是有长度限制的,所以还要做限制每次批量的数量,超过最大任务数,等下次执行
  76.                 if (i < MAX_TASK_NUM) {
  77.                     list.add(queue.poll());
  78.                 }
  79.             }
  80.             //拿到我们需要去数据库查询的特征,保存为集合
  81.             List<Request> userReqs = new ArrayList<>();
  82.             for (Request request : list) {
  83.                 userReqs.add(request);
  84.             }
  85.             //将参数传入service处理, 这里是本地服务,也可以把userService 看成RPC之类的远程调用
  86.             Map<String, Users> response = userService.queryUserByIdBatchQueue(userReqs);
  87.             for (Request userReq : userReqs) {
  88.                 // 这里再把结果放到队列里
  89.                 Users users = response.get(userReq.getRequestId());
  90.                 userReq.usersQueue.offer(users);
  91.             }
  92.         }, 10010, TimeUnit.MILLISECONDS);
  93.         //scheduleAtFixedRate是周期性执行 schedule是延迟执行 initialDelay是初始延迟 period是周期间隔 后面是单位
  94.         //这里我写的是 初始化后100毫秒后执行,周期性执行10毫秒执行一次
  95.     }
  96.     public Users queryUser(Long userId) {
  97.         Request request = new Request();
  98.         // 这里用UUID做请求id
  99.         request.requestId = UUID.randomUUID().toString().replace("-""");
  100.         request.userId = userId;
  101.         LinkedBlockingQueue<Users> usersQueue = new LinkedBlockingQueue<>();
  102.         request.usersQueue = usersQueue;
  103.         //将对象传入队列
  104.         queue.offer(request);
  105.         //取出元素时,如果队列为空,给定阻塞多少毫秒再队列取值,这里是3秒
  106.         try {
  107.             return usersQueue.poll(3000,TimeUnit.MILLISECONDS);
  108.         } catch (InterruptedException e) {
  109.             e.printStackTrace();
  110.         }
  111.         return null;
  112.     }
  113. }
  1. ...省略..
  2.     @Override
  3.     public Map<String, Users> queryUserByIdBatchQueue(List<UserWrapBatchQueueService.Request> userReqs) {
  4.         // 全部参数
  5.         List<Long> userIds = userReqs.stream().map(UserWrapBatchQueueService.Request::getUserId).collect(Collectors.toList());
  6.         QueryWrapper<Users> queryWrapper = new QueryWrapper<>();
  7.         // 用in语句合并成一条SQL,避免多次请求数据库的IO
  8.         queryWrapper.in("id", userIds);
  9.         List<Users> users = usersMapper.selectList(queryWrapper);
  10.         Map<Long, List<Users>> userGroup = users.stream().collect(Collectors.groupingBy(Users::getId));
  11.         HashMap<String, Users> result = new HashMap<>();
  12.         // 数据分组
  13.         userReqs.forEach(val -> {
  14.             List<Users> usersList = userGroup.get(val.getUserId());
  15.             if (!CollectionUtils.isEmpty(usersList)) {
  16.                 result.put(val.getRequestId(), usersList.get(0));
  17.             } else {
  18.                 // 表示没数据 , 这里要new,不然加入队列会空指针
  19.                 result.put(val.getRequestId(), new Users());
  20.             }
  21.         });
  22.         return result;
  23.     }
  24. ...省略...

小结

请求合并,批量的办法能大幅节省被调用系统的连接资源,本例是以数据库为例,其他RPC调用也是类似的道理。缺点就是请求的时间在执行实际的逻辑之前增加了等待时间,不适合低并发的场景。

代码地址
  • https://gitee.com/apple_1030907690/spring-boot-kubernetes/tree/v1.0.5

参考
  • https://www.cnblogs.com/oyjg/p/13099998.html

作者:愤怒的苹果ext

https://blog.csdn.net/baidu_19473529/article/details/124092081

  1. 公众号“Java精选”所发表内容注明来源的,版权归原出处所有(无法查证版权的或者未注明出处的均来自网络,系转载,转载的目的在于传递更多信息,版权属于原作者。如有侵权,请联系,笔者会第一时间删除处理!
  2. 最近有很多人问,有没有读者交流群!加入方式很简单,公众号Java精选,回复“加群”,即可入群!
  3. Java精选面试题(微信小程序):3000+道面试题,包含Java基础、并发、JVM、线程、MQ系列、Redis、Spring系列、Elasticsearch、Docker、K8s、Flink、Spark、架构设计等,在线随时刷题!
  4. ------ 特别推荐 ------
  5. 特别推荐:专注分享最前沿的技术与资讯,为弯道超车做好准备及各种开源项目与高效率软件的公众号,「大咖笔记」,专注挖掘好东西,非常值得大家关注。点击下方公众号卡片关注。
  6. 点击“阅读原文”,了解更多精彩内容!文章有帮助的话,点在看,转发吧!
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/article/detail/50230
推荐阅读
相关标签
  

闽ICP备14008679号