读书人

关于django多线程有关问题解决

发布时间: 2012-11-15 15:16:15 作者: rapoo

关于django多线程问题解决

Django自带的Development Server作为调试开发时的临时服务器是一个非常方便的工具,因为避免了在还没有代码之前就需要配置一大堆环境的麻烦。

但是此服务器也有一个不足的地方就是它是单线程模式的,也就是同时只能处理一个请求,当需要调试ajax应用时,这就显得非常麻烦了,因为这经常需要同时处理多个请求。

经过搜索,查找,简单的解决方法有以下两种:

1、修改Django自带的Development Server为多线程模式,这其实只需要改几行代码就可以了。(详见http://code.djangoproject.com/ticket/3357 )

打开文件 django/core/servers/basehttp.py

第10行:

from BaseHTTPServer import BaseHTTPRequestHandler, HTTPServer

修改为

from BaseHTTPServer import BaseHTTPRequestHandler

在所有的 import 语句之后加上以下代码:

Python代码 关于django多线程有关问题解决
  1. import BaseHTTPServer, SocketServer
  2. class HTTPServer(SocketServer.ThreadingMixIn, BaseHTTPServer.HTTPServer):
  3. def __init__(self, server_address, RequestHandlerClass=None):
  4. BaseHTTPServer.HTTPServer.__init__(self, server_address, RequestHandlerClass)

是的,这样就可以了。 但是他还有一个小问题就是它不是线程安全的。可能在请求很多时会出现数据不同步,当然,这一般不是问题,因为我们通常只在自己机器上调试时才用Development Server。

2、使用CherryPy 作为 Development Server

在安装好CherryPy 之后,将以下代码保存为文件runservercp.py

Python代码 关于django多线程有关问题解决
  1. from django.core.management.base import BaseCommand, CommandError
  2. from optparse import make_option
  3. import os, sys
  4. import cherrypy
  5. from cherrypy import wsgiserver
  6. class Command(BaseCommand):
  7. option_list = BaseCommand.option_list + (
  8. make_option('--noreload', action='store_false', dest='use_reloader', default=True,
  9. help='Tells Django to NOT use the auto-reloader.'),
  10. make_option('--adminmedia', dest='admin_media_path', default='',
  11. help='Specifies the directory from which to serve admin media.'),
  12. )
  13. help = "Starts a lightweight Web server for development."
  14. args = '[optional port number, or ipaddr:port]'
  15. # Validation is called explicitly each time the server is reloaded.
  16. requires_model_validation = False
  17. def handle(self, addrport='', *args, **options):
  18. import django
  19. from django.core.servers.basehttp import AdminMediaHandler
  20. from django.core.handlers.wsgi import WSGIHandler
  21. if args:
  22. raise CommandError('Usage is runservercp %s' % self.args)
  23. if not addrport:
  24. addr = ''
  25. port = '8000'
  26. else:
  27. try:
  28. addr, port = addrport.split(':')
  29. except ValueError:
  30. addr, port = '', addrport
  31. if not addr:
  32. addr = '127.0.0.1'
  33. if not port.isdigit():
  34. raise CommandError("%r is not a valid port number." % port)
  35. use_reloader = options.get('use_reloader', True)
  36. admin_media_path = options.get('admin_media_path', '')
  37. shutdown_message = options.get('shutdown_message', '')
  38. quit_command = (sys.platform == 'win32') and 'CTRL-BREAK' or 'CONTROL-C'
  39. def inner_run():
  40. from django.conf import settings
  41. from django.utils import translation
  42. print "\nValidating models..."
  43. self.validate(display_num_errors=True)
  44. print "\nDjango version %s, using settings %r" % (django.get_version(), settings.SETTINGS_MODULE)
  45. print "Development server is running at http://%s:%s/" % (addr, port)
  46. print "Quit the server with %s." % quit_command
  47. # django.core.management.base forces the locale to en-us. We should
  48. # set it up correctly for the first request (particularly important
  49. # in the "--noreload" case).
  50. translation.activate(settings.LANGUAGE_CODE)
  51. try:
  52. path = admin_media_path or django.__path__[0] + '/contrib/admin/media'
  53. handler = AdminMediaHandler(WSGIHandler(), path)
  54. cherrypy.config.update({
  55. 'server.socket_host': addr,
  56. 'server.socket_port': int(port),
  57. 'engine.autoreload_on': use_reloader
  58. })
  59. cherrypy.tree.graft(handler, '/')
  60. cherrypy.engine.start()
  61. cherrypy.engine.block()
  62. finally:
  63. if shutdown_message:
  64. print shutdown_message
  65. cherrypy.engine.exit()
  66. inner_run()

将该文件放到djang源代码目录 django\core\management\commands 下,这样运行命令

python manage.py runservercp

时就使用了CherryPy 作为服务器了,其中命令名是根据文件的名字而定的,因为该文件保存为runservercp.py,所以命令名为 runservercp。

另外放在django的源代码目录的好处是所有的django project都可以使用该命令使用到CherryPy 。

另外一种放置方式是新建一个app,在该app下新建package management,再在该package下建另一package commands,将该文件放在commands 目录下(可参考django\contrib\auth 应用),同时在settings.py 中的 INSTALLED_APPS 加上该app,那么该project 就可以使用该命令了。其他project不受影响。

读书人网 >编程

热点推荐