python – Flask:后台线程看到一个非空队列为空

前端之家收集整理的这篇文章主要介绍了python – Flask:后台线程看到一个非空队列为空前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
当我在uwsgi中运行Flask应用程序时,后台线程和应用程序功能查询相同队列的大小时会看到不同的值.

组件

>具有thread-safe queue的Flask应用程序.
> GET调用返回队列大小.
> POST调用向队列中添加一个元素.
>后台线程打印队列大小

问题

当应用程序是从shell使用python tester.py时,我得到预期的结果:

  1. 2014-06-07 14:20:50.677995 Queue size is: 0
  2. 127.0.0.1 - - [07/Jun/2014 14:20:51] "POST /addMessage/X HTTP/1.1" 200 -
  3. 2014-06-07 14:20:51.679277 Queue size is: 1
  4. 2014-06-07 14:20:52.680425 Queue size is: 1
  5. 2014-06-07 14:20:53.681566 Queue size is: 1
  6. 2014-06-07 14:20:54.682708 Queue size is: 1
  7. 127.0.0.1 - - [07/Jun/2014 14:20:55] "POST /addMessage/Y HTTP/1.1" 200 -
  8. 2014-06-07 14:20:55.687755 Queue size is: 2
  9. 2014-06-07 14:20:56.688867 Queue size is: 2

但是,当使用uwsgi执行应用程序时,我在日志中得到以下内容

  1. 2014-06-07 14:17:42.056863 Queue size is: 0
  2. 2014-06-07 14:17:43.057952 Queue size is: 0
  3. [pid: 9879|app: 0|req: 6/6] 127.0.0.1 () {24 vars in 280 bytes} [Sat Jun 7 14:17:43 2014] POST /addMessage/X => generated 16 bytes in 0 msecs (HTTP/1.1 200) 2 headers in 71 bytes (1 switches on core 0)
  4. 2014-06-07 14:17:44.059037 Queue size is: 0
  5. 2014-06-07 14:17:45.060118 Queue size is: 0
  6. [pid: 9879|app: 0|req: 7/7] 127.0.0.1 () {24 vars in 280 bytes} [Sat Jun 7 14:17:45 2014] POST /addMessage/X => generated 16 bytes in 0 msecs (HTTP/1.1 200) 2 headers in 71 bytes (1 switches on core 0)
  7. 2014-06-07 14:17:46.061205 Queue size is: 0
  8. 2014-06-07 14:17:47.062286 Queue size is: 0

当在uwsgi下运行时,后台线程看不到与应用程序相同的队列.这是为什么?我怎么能让这两个线程看同一个Queue对象?

更新

>即使将其作为Python脚本执行,我也看到不一致的行为:有时它不会记录消息(使用app.logger),我只能看到打印.这意味着线程正在运行,但它不能与app.logger做任何事情.

uwsgi .ini配置

  1. [uwsgi]
  2. http-socket = :9002
  3. plugin = python
  4. wsgi-file = /home/ubuntu/threadtest-uwsgi.py
  5. enable-threads = true
  6. workers = 1
  7. chdir = /home/ubuntu/thread-tester/thread_tester

  1. from flask import Flask,jsonify
  2. import Queue
  3. from threading import Thread
  4. import time
  5. import datetime
  6. import logging
  7. import sys
  8.  
  9. logging.basicConfig(stream=sys.stderr,format='%(asctime)s %(levelname)s - %(message)s')
  10.  
  11. app = Flask(__name__)
  12. messages = Queue.Queue()
  13.  
  14. def print_queue_size():
  15. while True:
  16. app.logger.debug("%s Queue size is: %d" % (datetime.datetime.now(),messages.qsize()))
  17. time.sleep(1)
  18.  
  19. t = Thread(target=print_queue_size,args=())
  20. t.setDaemon(True)
  21. t.start()
  22.  
  23. @app.route("/queueSize",methods=["GET"])
  24. def get_queue_size():
  25. return jsonify({"qsize": messages.qsize()}),200
  26.  
  27. @app.route("/addMessage/<message>",methods=["POST"])
  28. def add_message_to_queue(message):
  29. messages.put(message)
  30. return jsonify({"qsize": messages.qsize()}),200
  31.  
  32. if __name__ == "__main__":
  33. app.run(port=6000)

解决方法

Things to Know documenation page

uWSGI tries to (ab)use the Copy On Write semantics of the fork() call whenever possible. By default it will fork after having loaded your applications to share as much of their memory as possible. If this behavior is undesirable for some reason,use the lazy option. This will instruct uWSGI to load the applications after each worker’s fork(). Lazy mode changes the way graceful reloading works: instead of reloading the whole instance,each worker is reloaded in chain. If you want “lazy app loading”,but want to maintain the standard uWSGI reloading behavIoUr,starting from 1.3 you can use the lazy-apps option.

您的Flask应用程序在uWSGI启动时启动,然后分配一个工作进程.在分叉时,队列对象为空,不再与原始进程共享.线没有走.

尝试设置lazy-apps option以延迟加载Flask应用程序,直到工作人员启动.

猜你在找的Python相关文章