Tornado简介
Tornado全称Tornado Web Server,是一个用Python语言写成的Web服务器兼Web应用框架,由FriendFeed公司在自己的网站FriendFeed中使用,被Facebook收购以后框架在2009年9月以开源软件形式开放给大众。
特点:
-
作为Web框架,是一个轻量级的Web框架,类似于另一个Python web框架Web.py,其拥有异步非阻塞IO的处理方式。
-
作为Web服务器,Tornado有较为出色的抗负载能力,官方用nginx反向代理的方式部署Tornado和其它Python web应用框架进行对比,结果最大浏览量超过第二名近40%。
Tornado的特性
HTTP服务器
Tornado为了高效实现Comet/后端异步调用HTTP接口,是直接内嵌了HTTP服务器。
前端无需加apache / lighttpd / nginx等也可以供浏览器访问;但它并没有完整实现HTTP 1.1的协议,所以官方文档是推荐用户在生产环境下在前端使用nginx,后端反向代理到多个Tornado实例。
Tornado本身是单线程的异步网络程序,它默认启动时,会根据CPU数量运行多个实例;充分利用CPU多核的优势。
单线程异步
网站基本都会有数据库操作,而Tornado是单线程的,这意味着如果数据库查询返回过慢,整个服务器响应会被堵塞。
数据库查询,实质上也是远程的网络调用;理想情况下,是将这些操作也封装成为异步的;但Tornado对此并没有提供任何支持。
这是Tornado的设计,而不是缺陷。
一个系统,要满足高流量;是必须解决数据库查询速度问题的!
数据库若存在查询性能问题,整个系统无论如何优化,数据库都会是瓶颈,拖慢整个系统!
异步并不能从本质上提到系统的性能;它仅仅是避免多余的网络响应等待,以及切换线程的CPU耗费。
如果数据库查询响应太慢,需要解决的是数据库的性能问题;而不是调用数据库的前端Web应用。
对于实时返回的数据查询,理想情况下需要确保所有数据都在内存中,数据库硬盘IO应该为0;这样的查询才能足够快;而如果数据库查询足够快,那么前端web应用也就无将数据查询封装为异步的必要。
就算是使用协程,异步程序对于同步程序始终还是会提高复杂性;需要衡量的是处理这些额外复杂性是否值得。
如果后端有查询实在是太慢,无法绕过,Tornaod的建议是将这些查询在后端封装独立封装成为HTTP接口,然后使用Tornado内置的异步HTTP客户端进行调用。
开始 tornado 项目
# -*-coding: utf-8 -*- import tornado.web import tornado.ioloop # 引入 httpserver 模块 import tornado.httpserver class IndexHandler(tornado.web.RequestHandler): def get(self, *args, **kwargs): self.write("Hello Tornado!") if __name__ == '__main__': app = tornado.web.Application([ (r'/', IndexHandler), ]) # 直接使用 app 监听端口,最简单的写法,只能在单进程模式中使用 # app.listen(8000) # 实例化一个 http 服务器对象, 匹配 app 中的路由 httpServer = tornado.httpserver.HTTPServer(app) # 绑定端口,默认启动一个进程 # httpServer.listen(8000) # 绑定端口 httpServer.bind(8000) # 启动的进程的个数,默认开启一个进程,为 None 或者负数,也会开启对应的cpu核数个进程 httpServer.start(num_processes=5) # 一般不使用 tornado 提供的方法启动多个进程,使用手动方法启动进程,绑定不同的端口 # 1.每个子进程都会复制一份 ioloop 的实列,如果创建子进程前修改了 ioloop 会影响到多有的子进程 # 2.所有的进程都是一个命令启动的,无法做到在不停止服务的情况下修改代码 # 3.所有进程共享一个端口,很难进行分别监控 # 开始监听, 监听 epoll 中的请求 tornado.ioloop.IOLoop.current().start()