python中退出程序_python:优雅的退出程序或重启服务

在微服务中,使用任务队列有助于松耦合的设计,但有时,我们需要重启服务,但不能打断队列中正在进行的任务。

正确的做法是handle sigterm信号,具体代码如下:import sys

import argparse

import logging

import signal

import asyncio

class GracefulKiller:

kill_now = False

def __init__(self):

signal.signal(signal.SIGINT, self.exit_gracefully)

signal.signal(signal.SIGTERM, self.exit_gracefully)

def exit_gracefully(self,signum, frame):

self.kill_now = True

async def loop_task():

killer = GracefulKiller()

while 1:

print("ha") # 任务主体

if killer.kill_now:

break

await asyncio.sleep(2)

loop = asyncio.get_event_loop()

try:

loop.run_until_complete(loop_task())

finally:

loop.run_until_complete(loop.shutdown_asyncgens()) # see: https://docs.python.org/3/library/asyncio-eventloop.html#asyncio.AbstractEventLoop.shutdown_asyncgens

loop.close()

利用异步

python3.5 之后,有了比较完善的asyncio库和协程机制。对IO密集型任务,可以使用异步调用在单线程中实现“并发”。极大的增加任务吞吐。

想要让IO 任务并发,只需要使用支持asyncio的库(比如aiohttp),简单的loop.create_task就行。有时,需要限制后台任务的数量,在重启服务的时候,需要等待所有后台并发任务的完成。

此时消费者可以使用信号量进行控制。import argparse

import logging

import signal

import asyncio

class GracefulKiller:

kill_now = False

def __init__(self):

signal.signal(signal.SIGINT, self.exit_gracefully)

signal.signal(signal.SIGTERM, self.exit_gracefully)

def exit_gracefully(self,signum, frame):

self.kill_now = True

def pop_queue(task_queue):

try:

task_args = task_queue.pop(0)

print("dequeue args {}".format(task_args))

except:

task_args = None

return task_args

async def wait_tasks_done(semaphore, value):

while semaphore._value != value:

print("wait all task done!")

await asyncio.sleep(1)

# 处理单个异步任务

async def run_task(task_args, semaphore):

print("run_task {}".format(task_args))

await asyncio.sleep(3) # 模拟耗时

print("run_task {} done".format(task_args))

semaphore.release()

async def loop_task():

killer = GracefulKiller()

loop = asyncio.get_event_loop()

task_queue = [x for x in range(0, 10)] # 模拟一个任务队列

task_limit = 2 # 同时允许任务数量

semaphore = asyncio.Semaphore(value=task_limit, loop=loop)

while 1:

task_args = pop_queue(task_queue)

if task_args != None:

await semaphore.acquire()

loop.create_task(run_task(task_args, semaphore))

if killer.kill_now:

await wait_tasks_done(semaphore, task_limit)

break

loop = asyncio.get_event_loop()

try:

loop.run_until_complete(loop_task())

finally:

loop.run_until_complete(loop.shutdown_asyncgens())

loop.close()

输出:dequeue args 0

dequeue args 1

dequeue args 2

run_task 0

run_task 1

run_task 0 done

run_task 1 done

wait all task done!

run_task 2

wait all task done!

wait all task done!

run_task 2 done

上面的程序,无论何时重启,都将等待所有后台的任务完成。妈妈再也不用担心我重启服务被用户投诉了。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值