1. 第一种方法:启动的方式

gunicorn 启动的时候加上 --preload 参数,或者 preload_app=True ,默认为 False

preload的问题:
虽然这样可以使用scheduler创建代码只执行一次,但是问题也在于它只执行一次,重新部署以后如果用kill -HUP重启gunicorn,它并不会重启,甚至整个项目都不会更新。这是preload的副作用,除非重写部署脚本,完全重启应用。

2. 第二种方法:使用 锁 的方式来控制

from django.conf import settings
from apscheduler.schedulers.background import BackgroundScheduler
from django_apscheduler.jobstores import DjangoJobStore, register_events, register_job
from django_redis import get_redis_connection 

scheduler = BackgroundScheduler() 
scheduler.add_jobstore(DjangoJobStore(), "default") 
scheduler.remove_all_jobs() 

key = "lock" 
redis_cli = get_redis_connection(settings.REDIS_CACHE_ALIAS) 
data = redis_cli.get(key) 
if data is None: 
    redis_cli.set(key, "lock", ex=10) 
    register_events(scheduler) 
    scheduler.start()

启动时设置一个 key ,后面几个进程启动发现 key 已设置就不启动 apscheduler , 设置过期时间为启动后 key 的释放,这里只是一个简单实现思路,可以参考 并发锁 去实现。