php协程实现mysql异步_[转]向facebook学习,通过协程实现mysql查询的异步化

本文介绍了如何通过异步查询优化MySQL性能,以减少web网站的响应时间。Facebook早期通过hack PHP实现,但现在PHP5.5已内置支持。异步查询允许多条SQL语句并行执行,大幅缩短查询耗时。相比同步查询,异步查询在服务器端同时执行多条语句,从而提高效率。PHP的mysqli扩展提供了相应支持,结合yield和generator实现协程,保持代码习惯。然而,当MySQL服务器负载高时,大量并发可能会加重系统负担。示例代码展示了异步查询的执行流程,结果显示了性能提升。

前言

最近学习了赵海平的演讲,了解到facebook的mysql查询可以进行异步化,从而提高性能。由于facebook实现的比较早,他们不得不对php进行hack才得以实现。现在的php5.5,已经无需hack就可以实现了。

对于一个web网站的性能来说,瓶颈多半是来自于数据库。一般数据库查询会在某个请求的整体耗时中占很大比例。如果能提高数据库查询的效率,网站的整体响应时间会有很大的下降。如果能实现mysql查询的异步化,就可以实现多条sql语句同时执行。这样就可以大大缩短mysql查询的耗时。

异步为啥比同步快?

与异步查询相反的时同步查询。通常情况下mysql的query查询都是同步方式。下面我们对两种方式做下对比。对比的例子是,请求两次select sleep(1)。这条语句在mysql服务器端大概耗时1000ms。

同步方式的执行流程:

818c08a7217610f58197806b7f8c457a.png

第一步,向mysql服务器端发送第一次查询请求。大概耗时 1ms

第二步,mysql服务器端返回第一次查询的结果。大概耗时 1000ms

第三步,向mysql服务器再次发送请求。大概耗时 1ms

第四步,mysql服务器端返回第二次查询的结果。大概耗时 1000ms

同步的方式执行两次select sleep(1),大概耗时 2002ms。

异步方式的执行流程:

ee7d30ca5be7b394f53d7cafd3ce5585.png

第一步,向mysql服务器端发送第一次查询请求。大概耗时1ms

第二步,在等待第一次请求返回数据的同时,向服务器端发送第二次查询请求。大概耗时 1ms

第三步,接受mysql服务器端返回的两次查询请求。大概耗时 1000ms。

对比分析

异步查询比同步查询速度快,是因为多条查询语句在服务器端同时执行,大大缩短了服务器端的响应时间。并行一般情况下总比串行快嘛。sql语句执行时间越长,效果越明显。

如何实现mysql的异步查询?

要实现异步查询的关键是能把发送请求和接受返回数据分开。正好mysqlnd中提供了这个特性。

在mysqlnd中对应的方法是:

mysqlnd_async_query 发送查询请求

mysqlnd_reap_async_query 获取查询结果

mysqli扩展针对mysqlnd的这个特性做了封装,在调用query方法时,传入MYSQLI_ASYNC即可。

为啥使用协程?

查看了博文中的代码实现,是不是感觉写法和平时不一样?一般在项目当中,我们都是以function的形式去相互调用,function中包含了数据库查询。为了保持这个习惯,方便大家使用,因此引入了协程。在php5.5中正好提供了yield和generator,方便我们实现协程。示例代码如下:

在终端命令行方式执行结果如下:

从结果上我们可以看出执行流程是,先发了两次mysql查询,然后在接受数据库的返回数据。正常情况下,至少需要2000ms才能执行完毕。但是,real 0m1.016s,说明两次查询的耗时只有1016ms。

tips:以上代码只是示例代码,还有一些需要完善的地方。

注意

需要注意的是,如果mysql服务器本身负载很大,这种并行执行的方式就不一定是好的解决方法。因为,mysql服务端会为每个链接创建一个单独的线程进行处理。如果创建的线程数过多,会给系统造成负担。

参考资料

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值