1.什么是Tornado?
Tornado全称 Tornado Web Server,是一个用Python语言编写的Web服务器兼Web应用框架,由FriendFeed公司在自己的网站FriendFeed中使用,被Facebook收购之后在2009年9月以开源软件形式开发给大众。
特点:
- Tornado是一个轻量型的Web框架,拥有异步非阻塞IO的处理方式;
- 有较为出色的抗负载、处理高并发能力;
- tornado走的是少儿精的方向,注重的是性能优越,最出名的是异步非阻塞的设计方式;
C10K问题:
- C10K指的是同时处理1W客户端的并发请求
- tornado 在设计之初就考虑到了性能因素,旨在解决C10K的问题,这样的设计使tornado成为一个拥有非常高性能的解决方案(服务器与框架的集合体)
- epool(容器),内存映射、事件通知机制
补充:
并发:任务数大于CPU的内核数,通过操作系统的任务调度算法,实现用多个任务“一起”执行(实际上总有一些任务不在执行,因为切换任务的速度相当快,看上去像一起执行)。
并行:任务数小于等于CPU的内核数,即任务是真的一起执行。
2.Tornado的第一段代码
# coding:utf-8
from tornado.web import RequestHandler
import tornado.ioloop
import tornado.httpserver
class IndexHandler(RequestHandler):
""" 请求处理类 """
def get(self, *args, **kwargs):
# 对应用户的GET请求方式
self.write("hello, world!")
if __name__ == "__main__":
# 构建web应用实例
app = tornado.web.Application(
# 传入路由构建列表
[(r"/", IndexHandler)]
)
# 设置web应用的监听端口
app.listen(8000) # 此时服务器并未开启监听
# http_server = tornado.httpserver.HTTPServer(app)
# http_server.bind(8000) 绑定端口
# http_server.start(4) 开启多进程
# 运行当前的web应用实例
tornado.ioloop.IOLoop.current().start()
代码说明:
RequestHandler
封装了对应一个请求的所有信息和方法,write(响应信息)就是写响应信息的一个方法;对应每一种http请求方式(get、post等),把对应的处理逻辑写进同名的成员方法中(如对应get请求方式,就将对应的处理逻辑写在get()方法中),当没有对应请求方式的成员方法时,会返回“405: Method Not Allowed”错误。Application
Tornado Web框架的核心应用类,是与服务器对接的接口,里面保存了路由信息表,其初始化接收的第一个参数就是一个路由信息映射元组的列表;其listen(端口)方法用来创建一个http服务器实例,并绑定到给定端口(注意:此时服务器并未开启监听)。tornado.ioloop
tornado的核心io循环模块,封装了Linux的epoll和BSD的kqueue,是tornado高性能的基石。IOLoop.current():返回当前线程的IOLoop实例
IOLoop.start():启动IOLoop实例的I/O循环,同时服务器监听被打开
总结Tornado Web程序的编写思路
- 创建web应用实例对象,第一个初始化参数为路由映射列表
- 定义实现路由映射列表中的handler类
- 创建服务器实例,绑定服务器端口
- 启动当前线程的IOLoop
3.路由反解析
对于tornado的路由映射列表,实际上可以传入多个信息,如:
[
(r"/", IndexHandler),
(r"/subject", SubjectHandler, {"subject": "python"}),
(r"/score", ScoreHandler, {"python": 99}, name="score_url")
]
对于路由中的字典,会传入到对应的RequestHandler的initialize()方法中:
from tornado.web import RequestHandler
class SubjectHandler(RequestHandler):
def initialize(self, subject):
self.subject = subject
def get(self):
self.write(self.subject)
对于路由中的name字段,注意此时不能再使用元组的形式,而应使用tornado.web.url来构建,name是给该路由起的一个别名,可以通过调用RequestHandler.reverse_url(name)来获取该名称对应的url。
说明:因为考虑到url资源可能会更改,如果采用起别名的方式,当改变了url的定义方式时,在代码中也能使用reverse_url(name)来找到对应的url地址。
# coding:utf-8
import tornado.web
import tornado.ioloop
import tornado.httpserver
import tornado.options
from tornado.options import options, define
from tornado.web import url, RequestHandler
define("port", default=8000, type=int, help="run server on the given port.")
class IndexHandler(RequestHandler):
def get(self):
python_url = self.reverse_url("python_url")
self.write('<a href="%s">itcast</a>' %
python_url)
class ItcastHandler(RequestHandler):
def initialize(self, subject):
self.subject = subject
def get(self):
self.write(self.subject)
if __name__ == "__main__":
tornado.options.parse_command_line()
app = tornado.web.Application([
(r"/", Indexhandler),
url(r"/python", ItcastHandler, {"subject":"python"}, name="python_url")
],
debug = True)
http_server = tornado.httpserver.HTTPServer(app)
http_server.listen(options.port)
tornado.ioloop.IOLoop.current().start()