苏宁会员任务平台:基于异步化的性能优化实践

简介:

背景

苏宁会员任务平台是覆盖聚合电商、体育、金融、PPTV、直播、红孩子等各个业态,平台会实时获取用户的画像信息来计算用户在客群中的分布及画像属性,从而实时判断用户是否满足相关场景下任务,若满足相关场景以后可以领取任务下所有奖项;任务类型包含了订单红包、母婴、Super会员、直播、双签、金融升级存等等。在大促特别是双十一期间,任务中心产品对于各个业态的引流,会员的留存及转化来说是一个重要的工具。

问题

因任务平台业务逻辑复杂、实时性要求高,涉及多个外围系统服务及数据调用;一期系统上线后部分功能遇到性能问题,例如聚合页打开时间过长,首先聚合页上要展示用户能看到的任务列表,以及当前用户是否达到领取条件,其次每个任务需要展示的状态依赖于后台多种信息的聚合,包括不在有效时间范围内、当前时段库存、可供领取的总库存、领取频次等。复杂逻辑和实时要求导致TPS在上线压测的时候没有能够达到一个理想预期效果。

即将到来的”双十一”流量高峰, 可以预见会使得超过现有的任务系统的TPS的峰值, 从而导致任务系统在”双十一”的场景下很容易触碰到性能瓶颈,影响用户体验;因此需要对苏宁任务平台的核心功能做性能优化, 提升实时性复杂业务逻辑场景下的性能, 以便于应对任务平台的流量暴涨以及双十一流量高峰。

定位

现有的每个任务可能依赖于多个异构系统的服务或者数据,例如直播任务及订单任务来自于不同的系统的服务,并且有些场景是基于外围系统的数据进行逻辑计算,有些则是通过服务接口调用的方式。

代码示例:

 
public ResultDTO checkAndGetInfo() {
A a = getA(); B b = getB();
ResultDTO result = computeResult(a, b, c ...);
C c = getC(); ...... return resultDTO;
}

由于页面实时性要求高,逻辑复杂,对于某个任务是否展示需要调用多个外围接口,响应时间不可控,理论上根据任务的复杂性可能涉及多个客群,调用次数及响应时间不可控。性能主要在响应时间不可控。

某个任务状态要调用多个本地接口或者外围接口。

主要思路:异步,缓存,线程池

针对以上定位到位问题,考虑到实时调用外围接口的方案会导致响应时间不可控,采用NIO的思想,对整个调用链进行梳理,尽量异步化调用,同时增加适当过期时间的缓存,达到性能优化的目的。

在一期设计的时候已经从业务逻辑的角度做了拆分,将不同生命周期的逻辑异步化处理,例如奖励是通过kafka推送到奖励资源系统异步发放的。

上述从业务生命周期角度分析,通过切分业务流程,达到优化的方式已经不能满足系统性能需求,需要从技术上考虑更细粒度的异步化处理方式。

优化方案的选择及演进

Kilim

Kilim是一个java的协程框架,利用字节码技术编织技术将普通代码转化为支持协程的代码,当时是基于同步的思路下,想利用协程优化同步并发处理的能力。经过调研业界实践应用相对较少,因此考虑到项目开发周期等因素,没有采用Kilim方案。

Guava Listenable Future

JDK 5引入了Future模式。 Future接口是Java多线程Future模式的实现,在java.util.concurrent包中,可以来进行异步计算。

Future模式是多线程设计常用的一种设计模式。Future模式可以理解成:有一个任务,提交给了Future,Future替我完成这个任务。期间我自己可以去做任何想做的事情。一段时间之后,我就便可以从Future那儿取出结果。

 
ExecutorService executor = ...;
Future f = executor.submit(...);
f.get();

Future接口可以构建异步应用,但依然有其局限性。它很难直接表述多个Future 结果之间的依赖性。实际开发中,我们经常需要达成以下目的:

  1. 将多个异步计算的结果合并成一个

  2. 等待Future集合中的所有任务都完成

  3. Future完成事件(即,任务完成以后触发执行动作)

Future虽然可以实现获取异步执行结果的需求,但是它没有提供通知的机制,我们无法得知Future什么时候完成。

要么使用阻塞,在future.get()的地方等待future返回的结果,这时又变成同步操作。要么使用isDone()轮询地判断Future是否完成,这样会耗费CPU的资源。

Guava的Listenable Future对其做了改进,支持注册一个任务执行结束后回调函数。

 
ListenableFuture<String> listenableFuture =
listeningExecutor.submit(new Callable<String>() {
@Override
return "";
public String call() throws Exception { }
});

其中FutureCallback是一个包含onSuccess(V),onFailure(Throwable)的接口:

 
Futures.addCallback(ListenableFuture, new FutureCallback<Object>() {
public void onSuccess(Object result) { // do something on success
// do something on failure
} public void onFailure(Throwable thrown) { }
});

这也是一开始试验的方案,确定好了异步化的思路,自然联想到了增强版的Listenable Future,虽然在任务完成时可以回调函数通知,但是仍然是阻塞的,主线程仍然要等待异步线程完成任务通知。

Completable Future

Java8的CompletableFuture参考了Guava的ListenableFuture的思路,CompletableFuture能够将回调放到与任务不同的线程中执行,也能将回调作为继续执行的同步函数,在与任务相同的线程中执行。它避免了传统回调最大的问题,那就是能够将控制流分离到不同的事件处理器中。

CompletableFuture弥补了Future模式的缺点。在异步的任务完成后,需要用其结果继续操作时,无需等待。可以直接通过thenAccept、thenApply、thenCompose等方式将前面异步处理的结果交给另外一个异步事件处理线程来处理。

 
CompletableFuture completableFuture = new CompletableFuture();
completableFuture.whenComplete(new BiConsumer() { @Override
}); // complete the taskcompletableFuture.complete(new Object());//api method
public void accept(Object o, Object o2) { //handle complete }
completableFuture.thenAccept(Consumer c); //api method
completableFuture.thenApply(Function f); //api method

CompletableFuture 提出了CompletionStage的概念,代表异步计算过程中的某一个阶段,一个阶段完成以后可能会触发另外一个阶段。

一个阶段的计算执行可以是一个Function,Consumer或者Runnable。比如:

stage.thenApply(x -> square(x)).thenAccept(x -> System.out.print(x)).thenRun(() -> System.out.println());

一个阶段的执行可能是被单个阶段的完成触发,也可能是由多个阶段一起触发。

与Guava ListenableFuture相比,CompletableFuture不仅可以在任务完成时注册回调通知,而且可以指定任意线程,实现了真正的异步非阻塞。

Servlet 3.0

ed38d2676344a9ad841bf8765ce438457f103f05

传统Servlet 2.x web容器处理http请求时是为每一个请求分配一个线程,处理完请求再释放线程,如果请求处理的比较慢或者请求过多,就可能达到线程池达到上限,这时候后续的用户请求就会处于等待状态或者超时,这里用户请求和处理请求是一个线程,Servlet 3.0 开始提供了AsyncContext用来支持异步处理请求,主要是把请求线程和工作线程分开,将耗时的业务处理工作交给另外一个线程来完成。

 
@WebServlet(urlPatterns = "/servlet3",asyncSupported = true)
public class Servlet3 extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException { //在子线程中执行业务调用,并由其负责输出响应,主线程退出 AsyncContext ctx = request.startAsync();
public void doPost(HttpServletRequest request, HttpServletResponse response)
new Thread(new Executor(ctx)).start(); } throws ServletException, IOException { doGet(request, response); } } class Executor implements Runnable {
ServletRequest request = ctx.getRequest();
private AsyncContext ctx = null; public Executor(AsyncContext ctx){ this.ctx = ctx; } public void run(){ try { Thread.sleep(3000); ctx.dispatch("/index.jsp"); ctx.complete();
}
} catch (Exception e) { e.printStackTrace(); }
}

最终方案

最终选定Completable Future + Servlet 3.0的方案,前台web接口层采用Serlvet 3.0,后台服务层采用Completable Future。

验证

优化前压测数据:

ec35e04d831f77fe09494914ab78df99796491c6

图2:在访问聚合页200并发情况下的数据,TPS值3322,在用户并发量增加的时候,因依赖外部接口服务和原有的系统设计接口调用方法导致TPS基本不会随并发量的增加而提高。

优化后压测数据:

在访问聚合页100并发情况下的数据,TPS值5869,相对于优化之前的TPS有明显的提升。

66c33e7025f73cdc9a81329ea1ed864902bd5347

在访问聚合页150并发情况下的数据TPS值8581,在提高并发量的时TPS有显著的提高,说明优化后的效果很明显,也证实了优化方案是可行的。

3e6dae779ddcddc7151642baa15019bb92ac787a

总结

利用异步化来提升系统性能是一个整体、全链路的工作,仅仅依靠业务上的异步化,或者服务层的异步化远远不够,随着不同技术方案的选择及演进,对异步非阻塞模型有了更深入的了解之后,从前台用户请求到后端服务层处理,根据一整条链路的上每一层场景的不同,需要选取不同的异步化技术方案,才能达到系统整体性能提升的目的。


原文发布时间为:2018-11-4

本文作者:葛苏杰

本文来自云栖社区合作伙伴“Java杂记”,了解相关信息可以关注“Java杂记”。

相关文章
|
7月前
|
消息中间件 缓存 测试技术
企业微信针对百万级组织架构的客户端性能优化实践
本文主要分享的是企业微信在百对百万级大规模组织架构(后文简称大架构)时,是如何对客户端进行性能优化过程的,希望带给你启发。
48 0
|
2月前
|
数据采集 Web App开发 文字识别
高并发数据采集:Ebay商家信息多进程爬虫的进阶实践
高并发数据采集:Ebay商家信息多进程爬虫的进阶实践
|
2月前
|
小程序 前端开发 JavaScript
【经验分享】如何实现在支付宝小程序中开发营销组件
【经验分享】如何实现在支付宝小程序中开发营销组件
29 0
|
缓存 算法 数据库
在线直播系统源码,围绕缓存做了哪些工作?
在线直播系统源码,围绕缓存做了哪些工作?
|
负载均衡 应用服务中间件 nginx
一对一直播系统开发,解决技术难点是重点
在大量用户涌入平台的情况下,一对一直播系统开发还是要面临众多难题,只有解决这些技术难题,才能让一对一直播系统运行更加稳定。
|
运维 监控 测试技术
巧用友盟+U-APM 实现移动端性能优化—启动速度
移动端性能对用户体验、留存有着至关重要的影响,一个体验良好的应用,只有功能健全还不够,以下是我在性能优化上总结的几点:启动速度优化、流畅度优化、资源优化、内存优化、APK 体积优化。今天先聊聊,启动速度的那些事。
巧用友盟+U-APM 实现移动端性能优化—启动速度
|
视频直播 5G 开发工具
看直播软件源码,如何实现直播系统业务以及技术注意点分析
直播类app越来越受欢迎,互联网企业看到“直播+”的巨大潜力,搭建直播平台也成为大需求,而拥有一套完整的直播软件源码是开发直播软件的基础
看直播软件源码,如何实现直播系统业务以及技术注意点分析
盘点直播卖货系统开发要点功能
直播卖货系统开发作为行业内最脍炙人口的开发模式,凭借自身的优势在疫情期间稳定迅速的发展壮大起来,想要开发直播卖货系统,就要先了解需要具备哪些功能,本文就为大家简单盘点一下直播卖货系统的重要功能。
|
存储 XML 缓存
OTT端性能优化建设之本地缓存设计 | 《优酷OTT互联网大屏前端技术实践》第七章
目前,做2C业务的应用,更多强调SSR、客户端缓存以及PWA等,以实现首屏加载体验优化、秒开等性能指标,相比较而言,这些策略更加“综合”“强壮”,如果合理运用以及借助端能力,实现冷启动提速、首屏加载优化、秒开等不在话下。 但是笔者业务服务于“OTT端酷喵APP”前端业务,主要是酷喵APP的HTML5投放(目前更换使用Rax),而端内浏览器并不支持service worker(PWA),且受制于端及浏览器内核,并无zcache类似能力。至此,大写的无奈涌上心头,这种情况还能不能抢救一把?答案是:可以,localStorage迂回包抄方案。也介于此,本文方案诞生,虽不完美,但是终究有闪光所在。
1185 0
OTT端性能优化建设之本地缓存设计 | 《优酷OTT互联网大屏前端技术实践》第七章
简述购物直播系统开发的功能和优势有哪些
2020年的今天,直播已经深入我们生活的点点滴滴,通过直播我们不仅可以抓取我们的碎片化时间,丰富我们的娱乐方式,购物直播系统的兴起还改变了我们的购物方式,那么购物直播系统开发的功能和优势有哪些
简述购物直播系统开发的功能和优势有哪些

热门文章

最新文章