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()