问题:
- 由于发送短信验证码是耗时操作,如果发送短信的时候被阻塞,用户响应将会延迟
- 当后端产生了延迟,js中的回调函数也会产生延迟,会造成用户界面的倒计时延迟
- 效果就是用户点击了发送短信验证码,可能会产生倒计时效果不能及时显示
解决方案:
- 异步发送短信验证码
- 让发送短信和响应分开,将发送短信从主业务解耦出来
生产者消费者设计模式介绍
- 为了将发送短信从主业务中解耦出来,我们引入生产者消费者设计模式
- 项目生成发送短信验证码,缓存到消息队列,消费者读取消息队列中的发送短信消息并执行。
消息队列
消息队列是消息在传输过程中保存消息的容器,在此我们选择RabbitMQ作为消息队列。
此处省去了安装Erlang和安装RabbitMQ,我们需要注意的是:RabbitMQ提供的默认账户
用户名和密码:guest、guest
协议:amqp
地址:localhost
端口:5672
查看消息队中的信息:sudo rabbitatl list_queues
思考
- 消费者取到消息后,要消费掉
- 可能出先高并发的情况,需要补充多任务的方式执行
- 每一种耗时任务编写的生产者和消费者代码有重复
- 取到消息什么时候执行,怎么执行
Celery
介绍:
- 一个简单灵活可靠处理大量消息的分布式系统,可以在一台或者多台机器上运行
- 单个 Celery 进程每分钟可处理数以百万计的任务。
- 通过消息进行通信,使用消息队列
(broker)
在客户端和消费者之间进行协调。
安装:pip install -U Cerely
异步发送短信验证码的实现
1)首先,异步发送短信验证码是独立的。所以我们在项目目录下创建一个celery_tasks的包
2)在celery_tasks下创建main.py
celery_tasks.main.py中
# celery启动文件 from celery import Celery # 创建celery实例 celery_app = Celery('meiduo')
3)加载Celery配置
celery_tasks.config.py
# 指定消息队列的位置 broker_url= 'amqp://guest:guest@192.168.103.158:5672'
celery_tasks.main.py
# celery启动文件 from celery import Celery # 创建celery实例 celery_app = Celery('meiduo') # 加载celery配置 celery_app.config_from_object('celery_tasks.config')
定义发送短信任务
注册任务:celery_tasks.main.py
# celery启动文件 from celery import Celery # 创建celery实例 celery_app = Celery('meiduo') # 加载celery配置 celery_app.config_from_object('celery_tasks.config') # 自动注册celery任务 celery_app.autodiscover_tasks(['celery_tasks.sms'])
定义任务:celery_tasks.sms.tasks.py
# bind:保证task对象会作为第一个参数自动传入 # name:异步任务别名 # retry_backoff:异常自动重试的时间间隔 第n次(retry_backoff×2^(n-1))s # max_retries:异常自动重试次数的上限 @celery_app.task(bind=True, name='ccp_send_sms_code', retry_backoff=3) def ccp_send_sms_code(self, mobile, sms_code): """ 发送短信异步任务 :param mobile: 手机号 :param sms_code: 短信验证码 :return: 成功0 或 失败-1 """ try: send_ret = CCP().send_template_sms(mobile, [sms_code, constants.SMS_CODE_REDIS_EXPIRES // 60], constants.SEND_SMS_TEMPLATE_ID) except Exception as e: logger.error(e) # 有异常自动重试三次 raise self.retry(exc=e, max_retries=3) if send_ret != 0: # 有异常自动重试三次 raise self.retry(exc=Exception('发送短信失败'), max_retries=3) return send_ret
启动celery服务
1终端进入celery_tasks目录
2启动celery服务:celery -A celery_tasks.main worker -l info
调用任务
完成以上操作之后,调用ccp_send_sms_code就不会产生阻塞了