Tornado
文档:
https://tornado-zh-cn.readthedocs.io/zh_CN/latest/
github:
https://github.com/tornadoweb/tornado
介绍
Tornado是使用Python开发的全栈式(full-stack)Web框架和异步网络库,最早由4名Google前软件工程师(布雷特·泰勒)2007创办的Friendfeed(一个社交聚合网站)开发而来的。通过使用非阻塞IO,Tornado可以处理数以万计的开放连接,是long polling、WebSockets和其他需要为用户维护长连接应用的理想选择。
目前最新版本6.1, 我们实际项目开发是使用的不可能是最新版本,所以在此我们在tornado基础阶段所学所用的版本为6.0.
特点
-
开源的轻量级全栈式Web框架,提供了一整套完善的异步编码方案。
-
高性能
基于协程,底层就是基于asyio来实现的完整的协程调度
采用异步非阻塞IO处理方式,不依赖多进程或多线程
采用单进程单线程异步IO的网络模式,其高性能源于Tornado基于Linux的Epoll(UNIX为kqueue)的异步网络IO,具有出色的抗负载能力Tornado为了实现高并发和高性能,使用了一个
IOLoop
事件循环来处理
socket
的读写事件 -
WSGI全栈替代产品,Tornado把应用(Application)和服务器(Server)结合起来,既是WSGI应用也可以是WSGI服务,通俗来讲就是说,Tornado既是web服务器也是web框架,甚至可以通过Tornado替代uwsgi/gunicorn来运行Flask或者django框架
django,flask和tornado对比
内置功能模块来说: django > flask > tornado
使用入门门槛: django < flask < tornado
Tornado 可以被分为以下四个主要部分:
-
Web 框架 (包括用来创建 Web 应用程序的
RequestHandler
类, 还有很多其它支持的类). -
HTTP 客户端和服务器的实现 (
HTTPServer
和
AsyncHTTPClient
). -
异步网络库 (
IOLoop
和
IOStream
), 对 HTTP 的实现提供构建模块, 还可以用来实现其他协议. -
协程库 (
tornado.gen
) 让用户通过更直接的方法来实现异步编程, 而不是通过回调的方式.
安装
mkvirtualenv tornado_demo
pip install tornado==6.0.4
cd ~/Desktop
mkdir tdemo
cd tdemo
入门
项目基本运行
server.py
from tornado import ioloop # 异步网络底层模块 核心IO循环
from tornado import web # 核心web框架 包括Web框架大部分主要功能,包括RequestHandler和Application类
# 配置信息
settings = {}
# 视图类必须要直接或者间接继承于 web.RequestHandler
class Home(web.RequestHandler):
def get(self):
self.write("<h1>hello 锡洛!</h1>")
self.write("hello yanilo!")
def make_app():
# Application是tornado web框架的核心应用类,是于服务器对应的接口,里面保存了路由映射表s
return web.Application(
handlers=[ # 路由列表
# (r"uri路径", 视图类),
(r"/", Home),
],
**settings
)
if __name__ == "__main__":
# 创建web应用实例对象
app = make_app()
# 设置监听的端口和地址
app.listen(port=8888, address="0.0.0.0")
# ioloop,全局的tornado事件循环,是服务群器的引擎核心,start表示创建IO事件循环,等待客户连接
ioloop.IOLoop.current().start()
Tornado网络模型:
终端运行项目
server.py,代码:
from tornado import ioloop
from tornado import web
from tornado.options import define, options, parse_command_line
# 配置信息
settings = {}
# 视图类必须要直接或者间接继承于 web.RequestHandler
class Home(web.RequestHandler):
def get(self): # http请求
# 响应数据
self.write("<h1>hello python31</h1>")
self.write("hello tornado!") # 这里也是可以被执行的
return
self.write("hello world") # 只有在return以后,才不会被执行
def make_app():
# Application是tornado web框架的核心应用类,是与服务器对应的接口,里面保存了路由映射表s
return web.Application(
handlers=[ # 路由列表
# (r"uri路径", 视图类),
(r"/", Home),
],
# 配置数据
**settings
)
if __name__ == '__main__':
# 定义终端命令行参数
define(name="port", default=8888,type=int) # python 主程序.py --port=8888
define(name="host", default="127.0.0.1", type=str) # python 主程序.py --port=8888 --host=127.0.0.1
# 启动终端命令行参数解析方法
parse_command_line()
# 创建web应用实例对象
app = make_app()
# 设置监听的端口和地址
app.listen(port=options.port, address=options.host)
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环,等待客户端连接
ioloop.IOLoop.current().start()
开启调试模式
开启自动加载和调试模式,开启了debug模式以后, 项目在编辑python文件的时候自动重启项目并且在出现异常时显示错误跟踪信息
server.py, 代码:
from tornado import ioloop
from tornado import web
from tornado.options import define,options,parse_command_line
settings = {
"debug": True, # 开始debug模式
}
# 视图类必须要直接或者间接继承于 web.RequestHandler
class Home(web.RequestHandler):
def get(self): # http请求
# 响应数据
self.write("<h1>hello python31</h1>")
self.write("hello tornado!") # 这里也是可以被执行的
return
self.write("hello world") # 只有在return以后,才不会被执行
def make_app():
# Application是tornado web框架的核心应用类,是与服务器对应的接口,里面保存了路由映射表s
return web.Application(
handlers=[ # 路由列表
# (r"uri路径", 视图类),
(r"/", Home),
],
# 配置数据
**settings,
)
if __name__ == '__main__':
# 定义终端命令行参数
define(name="port", default=8888,type=int) # python 主程序.py --port=8888
define(name="host", default="127.0.0.1", type=str) # python 主程序.py --port=8888 --host=127.0.0.1
# 启动终端命令行参数解析方法
parse_command_line()
# 创建web应用实例对象
app = make_app()
# 设置监听的端口和地址
app.listen(port=options.port, address=options.host)
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环,等待客户端连接
ioloop.IOLoop.current().start()
路由拆分
代码:
from tornado import ioloop
from tornado import web
from tornado.options import define,options,parse_command_line
# 项目配置
settings = {
"debug": True, # 开启debug模式
}
# 视图类必须要直接或者间接继承于 web.RequestHandler
class Home(web.RequestHandler):
def get(self): # http请求
# 响应数据
self.write("<h1>hello! 锡落</h1>")
self.write("hello tornado!") # 这里也是可以被执行的
return
self.write("hello world") # 只有在return以后,才不会被执行
# 路由列表
urls = [
# (r"uri路径", 视图类),
(r"/", Home),
]
if __name__ == "__main__":
# 定义终端命令行参数
define(name="port",default=8888,type=int) # python 主程序.py --port=8888
define(name="host", default="127.0.0.1", type=str) # python 主程序.py --port=8888 --host=127.0.0.1
define(name="debug", default=False, type=bool) # # python 主程序.py --debug=True
# 启动终端命令行参数解析方法
parse_command_line()
# 创建web应用实例对象
# Application是tornado web框架的核心应用类,是与服务器对应的接口,里面保存了路由映射表
settings["debug"] = options.debug
app = web.Application(
urls,
**settings,
)
# 设置监听的端口和地址
app.listen(port=options.port,address=options.host)
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环,等待客户端连接
ioloop.IOLoop.current().start()
视图编写
在tornado中, tornado.web框架本身就默认提供了rest风格的API接口模式. 可以直接通过CBV(类视图)对外提供基本的http 视图接口.
from tornado import ioloop
from tornado import web
from tornado.options import define,options,parse_command_line
# 项目配置
settings = {
"debug": True, # 开启debug模式
}
# 视图类必须要直接或者间接继承于 web.RequestHandler
class Home(web.RequestHandler):
def get(self): # http请求
# 响应数据
self.write("<h1>hello! 锡落</h1>")
self.write("hello tornado!") # 这里也是可以被执行的
return
self.write("hello world") # 只有在return以后,才不会被执行
def post(self):
self.write("hello!post")
def put(self):
self.write("hello!put")
def patch(self):
self.write("hello!patch")
def delete(self):
self.write("hello!delete")
# 路由列表
urls = [
# (r"uri路径", 视图类),
(r"/", Home),
]
if __name__ == "__main__":
# 定义终端命令行参数
define(name="port",default=8888,type=int) # python 主程序.py --port=8888
define(name="host", default="127.0.0.1", type=str) # python 主程序.py --port=8888 --host=127.0.0.1
define(name="debug", default=False, type=bool) # # python 主程序.py --debug=True
# 启动终端命令行参数解析方法
parse_command_line()
# 创建web应用实例对象
# Application是tornado web框架的核心应用类,是与服务器对应的接口,里面保存了路由映射表
settings["debug"] = options.debug
app = web.Application(
urls,
**settings,
)
# 设置监听的端口和地址
app.listen(port=options.port,address=options.host)
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环,等待客户端连接
ioloop.IOLoop.current().start()
多进程模式
from tornado import ioloop
from tornado import web, httpserver
from tornado.options import define, options, parse_command_line
settings = {
"debug": True, # 开始debug模式
}
# 视图类必须要直接或者间接继承于 web.RequesHandler
class Home(web.RequestHandler):
def get(self): # http请求
self.write("<h1>hello! 锡落</h1>")
self.write("hello tornado!") # 这里也可以被执行
return
self.write("hello tornado!!!") # 只有在return以后,才不会被执行
# 路由列表
urls = [
# (r"url路径", 视图类),
(r"/", Home),
]
if __name__ == '__main__':
# 定义终端命令行参数
define(name="port",default=8888,type=int) # python 主程序.py --port=8888
define(name="host", default="127.0.0.1", type=str) # python 主程序.py --port=8888 --host=127.0.0.1
define(name="debug", default=False, type=bool) # # python 主程序.py --debug=True
# 启动终端命令行参数解析方法
parse_command_line()
# 创建web应用实例对象
# Application是tornado web框架的核心应用类,是与服务器对应的接口,里面保存了路由映射表
settings["debug"] = options.debug
app = web.Application(
urls,
**settings,
)
# 设置监听的端口和地址
# 启动http多进程服务
server = httpserver.HTTPServer(app)
server.listen(port=options.port,address=options.host)
server.start(num_processes=0) # 0表示CPU的核心数+1, 设置其他数字责n+1
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环,等待客户端连接
ioloop.IOLoop.current().start()
请求与响应
请求
tornado.httputil.HTTPServerRequest
server.py,代码:
from tornado import ioloop
from tornado import web
from tornado.options import define, options, parse_command_line
# 项目配置
settings = {
"debug": True, # 开启debug模式
}
# 视图类必须要直接或者间接继承于 web.RequestHandler
class Home(web.RequestHandler):
def get(self): # http请求方法
# 响应数据
# print(self.settings) # 配置信息
# print(self.request) # 获取http请求处理的实例对象
# # HTTPServerRequest(protocol='http', host='127.0.0.1:8000', method='GET', uri='/', version='HTTP/1.1', remote_ip='127.0.0.1')
print("通信协议: ",self.request.protocol) # 协议
print("请求方法: ",self.request.method) # Http请求方法
print("uri地址: ",self.request.uri) # uri地址
print("url地址: ",self.request.full_url()) # 完整url地址
print("协议版本: ",self.request.version) # 协议版本
print("请求头: ")
print(self.request.headers) # 请求头 HTTPHeaders
print("地址端口: ", self.request.host) # 地址端口
self.write("hello!get") # 这里也是可以被执行的
def post(self):
# print("请求体: ", self.request.body) # 请求体[原始数据]
# import json
# body = json.loads(self.request.body.decode())
# print(body) # {'jsonrpc': '2.0', 'id': 1, 'method': 'Live.stream.list', 'params': {}}
# print("上传文件: ",self.request.files)# 上传文件
# print("cookie信息: ")
# print(self.request.cookies) # cookie信息
# print("当前客户端的IP地址: ", self.request.remote_ip) # 客户端IP地址
print(self.request.request_time()) # 请求处理的花费时间
self.write("hello!post")
def put(self):
# 开发中很少使用
print("查询字符串参数列表: ",self.request.query_arguments) # 查询字符串参数列表
print("请求体参数列表: ",self.request.body_arguments) # 请求体参数列表
self.write("hello!put")
# 路由列表
urls = [
# (r"uri路径", 视图类),
(r"/", Home),
]
if __name__ == "__main__":
# 定义终端命令行参数
define(name="port", default=8888, type=int) # python 主程序.py --port=8888
define(name="host", default="127.0.0.1", type=str) # python 主程序.py --port=8888 --host=127.0.0.1
define(name="debug", default=False, type=bool) # # python 主程序.py --debug=True
parse_command_line()
# Application是tornado web框架的核心应用类,是与服务器对应的接口,里面保存了路由映射表
app = web.Application(
urls,
**settings,
)
# 设置监听的端口和地址
# 启动http多进程服务
app.listen(port=options.port,address=options.host)
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环,等待客户端连接
ioloop.IOLoop.current().start()
接收查询字符串与请求体
server.py,代码:
from tornado import ioloop
from tornado import web
# 项目配置
settings = {
"debug": True, # 开启debug模式
}
# 视图类必须要直接或者间接继承于 web.RequestHandler
class Home(web.RequestHandler):
def get(self):
# 获取查询字符串参数
print(self.request.query ) # uname=xiaohuihui&age=6&lve=game&lve=shopping
print(self.request.query_arguments ) # {'uname': [b'xiaohuihui'], 'age': [b'6'], 'lve': [b'game', b'shopping']}
print(self.get_query_argument("lve")) # 获取查询字符串中指定参数的最后一个值, shopping
print(self.get_query_arguments("lve")) # 获取查询字符串中指定参数的所有值, ['game', 'shopping']
self.write("hello!get")
def post(self):
"""获取请求体中数据"""
print(self.request.body)
print(self.request.body_arguments) # {'uname': [b'xiaobai'], 'age': [b'21'], 'lve': [b'swimming', b'game']}
print(self.get_body_argument("lve")) # 获取http请求体中指定参数的最后一个值, game
print(self.get_body_arguments("lve")) # 获取http请求体中指定参数的所有值 ['swimming', 'game']
self.write("hello!post")
def put(self):
"""获取请求体和查询字符串参数"""
# 既可以提取查询字符串的参数,也会提取请求体中的参数
print(self.request.arguments) # {'uname': [b'xiaohuihui', b'xiaobai'], 'age': [b'6', b'21'], 'lve': [b'game', b'shopping', b'swimming', b'game1']}
print(self.get_argument("lve",None)) # game1
print(self.get_arguments("lve")) # ['game', 'shopping', 'swimming', 'game1']
# 路由列表
urls = [
# (r"uri路径", 视图类),
(r"/", Home),
]
if __name__ == "__main__":
# Application是tornado web框架的核心应用类,是与服务器对应的接口,里面保存了路由映射表
app = web.Application(
urls,
**settings,
)
# 设置监听的端口和地址
# 启动http多进程服务
app.listen(port=8000,address="0.0.0.0")
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环,等待客户端连接
ioloop.IOLoop.current().start()
浏览器:
http://127.0.0.1:8888/?name=xiaoming&name=xiaohong
接收路由参数
server.py,代码:
from tornado import ioloop
from tornado import web
# 项目配置
settings = {
"debug": True, # 开启debug模式
}
# 视图类必须要直接或者间接继承于 web.RequestHandler
class Home(web.RequestHandler):
def get(self,cat,id): # 路由参数的形参,分别匹配路由中对应小括号中的url地址内容
self.write("hello!cat=%s,id=%s" % (cat,id))
class Index(web.RequestHandler):
def get(self,cat,id):
self.write("hello!cat=%s,id=%s" % (cat, id))
# 路由列表
urls = [
# (r"uri路径", 视图类),
# (r"/(参数1的正则)/(参数2的正则).html", Home), # 位置参数
(r"/([0-9]+)-([0-9]+).html", Home), # 位置参数
# (r"/index/(?P<参数名>正则).html", Index), # 命名参数
(r"/index/(?P<id>[0-9]+)-(?P<cat>[0-9]+).html", Index), # 命名参数
]
if __name__ == "__main__":
# Application是tornado web框架的核心应用类,是与服务器对应的接口,里面保存了路由映射表
app = web.Application(
urls,
**settings,
)
# 设置监听的端口和地址
# 启动http多进程服务
app.listen(port=8000,address="0.0.0.0")
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环,等待客户端连接
ioloop.IOLoop.current().start()
响应
server.py,代码:
from tornado import ioloop
from tornado import web
from tornado import autoreload
from tornado.options import define, options, parse_command_line
settings = {
'debug': True,
}
define("port", default=8888, type=int, help="设置监听端口号,默认为8888")
from datetime import datetime
class Home(web.RequestHandler):
def set_default_headers(self):
self.set_header("time", int(datetime.now().timestamp()))
def get(self):
def get(self):
# write 会自动识别
# self.write("<h1>hello world</h1>") # 响应html文档
# self.write({"msg":"ok"}) # 响应json数据
# 注意,json的数据格式也可以是列表,tornado中默认不支持返回list,所以如果返回list,则需要重写write
# self.write([1,2,3])
# 重新设置响应头的内容 set_header[修改]
self.set_header("Content-Type","text/json; charset=gbk")
# 自定义响应头 add_header[添加]
self.add_header("Company","OldBoyEdu")
self.add_header("Server","OldBoyEduServer/1.0")
# self.clear_header("Server") # 从响应头中删除指定名称的响应头信息
def post(self):
# self.set_status(404,"No User") # 第二个参数表示响应描述,如果不设置,则显示原来对应的
# self.send_error(500,reason="服务器炸了!")
self.send_error(404, msg="服务器炸了!", info="快报警") # 要使用send_error必须先声明send_error方法
def write_error(self, status_code, **kwargs):
self.write("<h1>完蛋啦...</h1>")
self.write("<p>错误信息:%s</p>" % kwargs["msg"])
self.write("<p>错误描述:%s</p>" % kwargs["info"])
def put(self):
# 页面响应
self.redirect("http://www.oldboyedu.com")
# 设置路由列表
urls = [
(r"/", Home),
]
if __name__ == "__main__":
# 创建应用实例对象
parse_command_line()
app = web.Application(urls, **settings)
# 设置监听的端口和地址
app.listen(options.port)
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环
ioloop.IOLoop.current().start()
cookie
基本使用
设置cookie self.set_cookie(name, value)
获取cookie self.get_cookie(name)
# 获取和设置cookie
self.set_cookie(name, value)
self.get_cookie(name)
# 获取和设置cookie[加密]
settings = {cookie_secret:"u3h7AQnM2WdTL1o"}
self.set_secure_cookie(name, value)
self.get_secure_cookie(name)
# 删除cookie
self.clear_cookie(name)
# 清空cookie
self.clear_all_cookie()
tornado没有提供session操作,如果需要使用到session可以自己实现或者引入第三方模块。
server.py,代码:
from tornado import web
from tornado import ioloop
settings = {
"debug": True,
}
from datetime import datetime
class Home(web.RequestHandler):
def get(self):
# 设置cookie
self.set_cookie("uname","xiaoming",expires=int(datetime.now().timestamp())+10)
self.write("set cookie")
class Index(web.RequestHandler):
def get(self):
# 获取cookie
uname = self.get_cookie("uname","")
self.write("uname=%s" % uname)
urls = [
(r"/cookie/set", Home),
(r"/cookie/get", Index),
]
if __name__ == '__main__':
app = web.Application(urls,**settings)
app.listen(port=8888)
ioloop.IOLoop.current().start()
加密使用
设置cookie self.set_secure_cookie(name,value)
获取cookie self.get_secure_cookie(name)
删除cookie self.clear_cookie(name)
清空cookie self.clear_all_cookie()
server.py,代码:
from tornado import web
from tornado import ioloop
settings = {
"debug": True,
# import base64, uuid
# base64.b64encode(uuid.uuid4().bytes + uuid.uuid4().bytes)
"cookie_secret": "WO+JNAJ3QZyOe4SMVXZpXAt3uG9hoU0UokoCBeYn1Y4="
}
from datetime import datetime
class Home(web.RequestHandler):
def get(self):
self.set_secure_cookie("name","xiaoming",expires=int(datetime.now().timestamp())+30)
self.set_secure_cookie("age","16",expires=int(datetime.now().timestamp())+30)
self.write("set cookie")
class Index(web.RequestHandler):
def get(self):
# 获取cookie[加密]
age = self.get_secure_cookie("age")
name = self.get_secure_cookie("name")
if age is not None:
age = age.decode()
if name is not None:
name = name.decode()
self.write("age=%s,name=%s" % (age,name))
class Page(web.RequestHandler):
def get(self):
"""删除cookie"""
# self.clear_cookie("age") # 删除指定名称的cookie[不管是否有加密]
self.clear_all_cookies() # 删除所有cookie[慎用]
self.write("del cookie")
urls = [
(r"/cookie/set", Home),
(r"/cookie/get", Index),
(r"/cookie/del", Page),
]
if __name__ == '__main__':
app = web.Application(urls,**settings)
app.listen(port=8888)
ioloop.IOLoop.current().start()
注意:
- tornado没有提供session操作,如果需要使用到session可以自己实现或者引入第三方模块。
- cookie的删除, 是不管是否加密的.
静态文件
from tornado import web
from tornado import ioloop
import os
settings = {
'debug': True,
# 静态文件保存路径
"static_path": os.path.join(os.path.dirname(__file__), 'static'),
# 静态文件url地址前缀
"static_url_prefix":"/static/", # 必须前后有斜杠
# 提供静态文件访问支持的视图类
"static_handler_class": web.StaticFileHandler,
}
class Home(web.RequestHandler):
def get(self):
# 项目中使用join拼凑路径时,必须注意第二个参数,必能以斜杠开头,会出现路径穿越(路径穿透)问题
path = os.path.join(os.path.dirname(__file__),"/static")
self.write("path=%s" % path)
urls = [
(r"/", Home),
# 上面settings中关于静态文件的配置,主要是提供给Application应用对象进行初始化生成下面路由时候使用到的。
# (r"/static/(.*)", web.StaticFileHandler, {"path": os.path.join(os.path.dirname(__file__), 'static')}),
]
if __name__ == '__main__':
app = web.Application(urls,**settings)
app.listen(port=8888)
ioloop.IOLoop.current().start()
页面响应
加载template模板
from tornado import web
from tornado import ioloop
import os
settings = {
'debug': True,
"template_path": os.path.join(os.path.dirname(__file__), 'templates'),
}
class Home(web.RequestHandler):
def get(self):
self.render("index.html",data={"message":"hello world"})
urls = [
(r"/", Home),
]
if __name__ == '__main__':
app = web.Application(urls,**settings)
app.listen(port=8888)
ioloop.IOLoop.current().start()
templates/index.html,代码:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<p>{{data["message"]}}</p>
tornado默认内置了一套非常强大的模板引擎<br>
这套模板引擎是基于jinja2模板引擎的基础上进行了改造而成的。<br>
当然jinja2是基于django的DTL模板引擎基础上改造而成的。<br>
所以flask和tornado进行比较的时候,从来不提tornado抄袭模板引擎这个事,反而会和django去比较模板引擎的问题。
</body>
</html>
路由进阶
路由语法和参数
在路由列表的路由成员中,我们一共可以设置4个参数
url(r”/uri路径”, 视图类, {“参数名”:“参数值”}, name=“路由别名”)
server.py
from tornado import web
from tornado import ioloop
settings = {
'debug': True,
}
class Home(web.RequestHandler):
def initialize(self,company)-> str:
# initialize 初始化方法[钩子方法]
self.company = company
def get(self):
print(self.company)
print("uri路径:%s" % self.reverse_url("home") ) # 对路由别名进行 反解析
self.write("hello,get")
def post(self):
print(self.company)
from tornado.web import url
urls = [
# (r"/", Home), # 这个格式的路由其实是简写模式, 在tornaodo.web中内部中最终由 _ApplicationRouter 的 Rule来进行封装和匹配路由和视图的关系
# url(pattern=路由uri地址, handler=视图类,kwargs=提供给视图类的公共参数,name="路由别名,用于反解析"),
url(pattern=r"/abc", handler=Home,kwargs={"company":"OldBoyEdu"},name="home"),
]
if __name__ == '__main__':
app = web.Application(urls,**settings)
app.listen(port=8888)
ioloop.IOLoop.current().start()
视图进阶
视图中内置的钩子方法
在tornado提供的视图类中,我们除了可以编写客户端http请求对应名称的视图方法和初始化方法
initialize
以外,还提供了一个预处理方法
prepare
和
on_finish
,prepare方法会在http请求方法执行之前先执行,
on_finish
会在http响应完成时进行。
server.py,代码:
from tornado import ioloop
from tornado import web
from tornado.httpserver import HTTPServer
from tornado.options import define, options, parse_command_line
from tornado.web import url
settings = {
'debug': True,
}
define("port", default=8888, type=int, help="设置监听端口号,默认为8888")
class Home(web.RequestHandler):
def prepare(self):
print(self.request.headers.get("Content-Type"))
if self.request.headers.get("Content-Type","").startswith("application/json"):
import json
self.json_data = json.loads(self.request.body)
self.is_json = True
else:
self.is_json = False
self.json_data = None
def post(self):
if self.is_json:
print(self.json_data)
self.write("hello,post")
def on_finish(self):
print("请求处理结束!可以在此进行收尾工作,例如:记录日志等")
# 设置路由列表
urls = [
(r"/", Home),
]
if __name__ == "__main__":
# 创建应用实例对象
parse_command_line()
app = web.Application(urls, **settings)
server = HTTPServer(app)
# 设置监听的端口和地址
server.listen(options.port)
server.start(1)
ioloop.IOLoop.current().start()
视图方法调用顺序
from tornado import ioloop
from tornado import web
from tornado.httpserver import HTTPServer
from tornado.options import define, options, parse_command_line
from tornado.web import url
settings = {
'debug': True,
}
define("port", default=8888, type=int, help="设置监听端口号,默认为8888")
class Home(web.RequestHandler):
def initialize(self):
print("initialize执行了")
def prepare(self):
print("prepare执行了")
def set_default_headers(self):
print("set_default_headers执行了")
def get(self):
self.write("hello,get")
print("视图http方法执行了")
# self.send_error(200,msg="注意:丢炸弹了") # 此处抛出错误
def write_error(self, status_code, **info):
print("write_error执行了,msg=%s" % info["msg"])
def on_finish(self):
print("on_finish执行了")
# 设置路由列表
urls = [
(r"/", Home),
]
if __name__ == "__main__":
# 创建应用实例对象
parse_command_line()
app = web.Application(urls, **settings)
server = HTTPServer(app)
# 设置监听的端口和地址
server.listen(options.port)
server.start(1)
ioloop.IOLoop.current().start()
当视图中没有任何异常时,执行顺序:
set_defautl_headers()
initialize()
prepare()
视图http方法()
on_finish()
当视图中抛出异常错误时,执行顺序:
set_default_headers()
initialize()
prepare()
视图http方法()
set_default_headers()
write_error()
on_finish()
冲刷缓存
在前面的学习中,我们使用了
self.write()
来完成数据的响应。
事实上,在tornado提供的视图操作中,视图中提供了一个 _write_buffer列表用于暂时缓存提供给客户端的数据, 这个 _write_buffer就是输出缓冲区
self.write()
本质上来说是将chunk数据块写到输出缓冲区中。所以才出现在视图中多次调用
self.write()
输出数据的情况,因为self.write根本没有输出数据,而是把数据写入到了输出缓冲区里面. 如果没有其他操作干预的情况下,则视图方法处理完成以后,会讲输出缓冲区中所有的数据冲刷出来响应给客户端。
除了
self.write()
方法以外,tornado还提供了2个方法用于在视图中冲刷缓存数据到客户端的。
self.flush()
立刻把数据从输出缓冲区冲刷出去。
self.finish()
立刻把数据从输出缓冲区冲刷出去。但是与self.flush()不同的是, self.finish()执行了以后, 后面的所有输出调用都不在支持.也就不能返回给客户端。
server.py,代码:
from tornado import ioloop
from tornado import web
from tornado.httpserver import HTTPServer
from tornado.options import define, options, parse_command_line
settings = {
'debug': True,
}
define("port", default=8888, type=int, help="设置监听端口号,默认为8888")
class Home(web.RequestHandler):
def get(self):
import time
self.write("hello,get1")
self.flush()
time.sleep(3)
self.write("hello,get2")
self.flush()
time.sleep(3)
self.write("hello,get3")
self.flush()
self.finish("这里一般不写任何内容,表示视图处理结束")
self.write("hello,get4")
# 设置路由列表
urls = [
(r"/", Home),
]
if __name__ == "__main__":
# 创建应用实例对象
parse_command_line()
app = web.Application(urls, **settings)
server = HTTPServer(app)
# 设置监听的端口和地址
server.listen(options.port)
server.start(1)
ioloop.IOLoop.current().start()
用户认证
tornado提供了装饰器tornado.web.authenticated与视图内置方法
get_current_user
允许我们轻松的实现用户认证功能。
装饰器authenticated依赖于请求处理类中的self.current_user属性来进行判断用户是否通过认证,如果self.current_user值为假(None、False、0、””等),任何GET或HEAD请求都将把访客重定向到settings配置中login_url设置的URL,而非法用户的POST请求将返回HTTPError(403)异常, Forbidden。
server.py,代码:
from tornado import web
from tornado import ioloop
settings = {
'debug': True,
# 登录页面的url地址
"login_url": r"/login"
}
from tornado.web import authenticated
class HttpRequest(web.RequestHandler):
def get_current_user(self):
username = self.get_argument("username", "")
password = self.get_argument("password", "")
if username == "root" and password == "123":
return username
class Home(HttpRequest):
@authenticated
def get(self):
self.write("hello,用户个人中心")
@authenticated
def post(self):
self.write("hello,用户中心")
class UserLogin(web.RequestHandler):
def get(self):
self.write("登录页面")
urls = [
(r"/", Home),
(settings["login_url"], UserLogin),
]
if __name__ == '__main__':
app = web.Application(urls,**settings)
app.listen(port=8888)
ioloop.IOLoop.current().start()
先访问:http://127.0.0.1:8888/?username=root&password=123
再访问:http://127.0.0.1:8888/
模板语法
基本语法
变量、表达式与自定义函数
server.py,代码:
from tornado import ioloop
from tornado import web
from tornado.options import define, options, parse_command_line
import os
settings = {
'debug': True,
# 静态文件保存路径
"static_path": os.path.join(os.path.dirname(__file__), 'static'),
# 静态文件url地址前缀
"static_url_prefix":"/static/", # 必须前后有斜杠
"template_path": os.path.join(os.path.dirname(__file__), 'templates'),
}
define("port", default=8888, type=int, help="设置监听端口号,默认为8888")
def money_format(data):
return "%.2f" % data
import time
class Home(web.RequestHandler):
def get(self):
data = {
"name":"xiaoming",
"money": 100,
"address":["北京市","昌平区","白沙路地铁站"]
}
info = {"name":"xiaoming"}
self.render("index.html",time=time,info=info,money_format=money_format,**data)
# 设置路由列表
from tornado.web import url
urls = [
url(r"/", Home,{}, name="home"),
]
if __name__ == "__main__":
# 创建应用实例对象
parse_command_line()
app = web.Application(urls, **settings)
# 设置监听的端口和地址
app.listen(options.port)
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环
ioloop.IOLoop.current().start()
templates/index.html,代码:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
{# 模板注释 #}
<p>name={{name}}</p>
<p>name={{info['name']}}</p>
<p>{{"-".join(address)}}</p>
<p>{{money_format(money)}}</p>
</body>
</html>
控制语句
# 判断
{% if ... %}
{% elif ... %}
{% else ... %}
{% end %}
# 遍历
{% for ... in ... %}
{% end %}
# 循环
{% while ... %}
{% end %}
内置标签或函数
内置标签
# 导包
{% from ... import ... %}
{% import ... %}
# 加载其他模板
{% include ... %}
# 输出原始数据
{% raw ... %}
# 语句/局部变量
{% set 变量名=变量值 %}
# 异常处理
{% try %}...{% except %}...{% else %}...{% finally %}...{% end %}
# 模板继承
{% extends *filename* %}
{% block 模板块名称} {% end %}
内置函数
# 输出转义数据,tornado在配置中允许通过autoescape=None设置全局转义
{{ escape(text) }}
# 静态文件存储路径
{{ static_url("style.css") }}
# 路由反解析
reverse_url("路由别名")
# CSRF防范机制,CSRF也叫XSRF
# tornado开启csrf必须在配置中进行设置 xsrf_cookies = True
# 补充:
# 在前后端分离项目中,客户端可以通过cookie来读取XSRFToken,cookie名称为_xsrf,请求头必须名称:X-XSRFToken
# 在视图方法中可以通过 self.xsrf_token 来获取 XSRFTokentoken
{% module xsrf_form_html() %}
内置变量
# 客户端请求对象
request
数据库
与Django框架相比,Tornado没有自带ORM,对于数据库需要自己去适配。我们使用MySQL数据库。
在Tornado3.0版本以前提供tornado.database模块用来操作MySQL数据库,而从3.0版本开始,此模块就被独立出来,作为torndb包单独提供。torndb只是对MySQLdb的简单封装,不支持Python 3。所以如果在当前版本中使用torndb进行数据库操作,需要修改源代码,所以在此,我们使用pymysql。
项目中如果要使用ORM,可以使用SQLAlchemy,但开发中,很少有人这么使用.
同时,tornado强大的地方在于其异步非阻塞,所以我们后面关于数据库操作,不管是mysql, mongodb还是redis基本都是异步读写操作。
MySQL
pip install pymysql
mysql.py,代码:
import pymysql
class MySQL(object):
def __init__(self, host, user, pwd, name):
self.host = host
self.user = user
self.pwd = pwd
self.name = name
self.data = None
self.last_sql = None
def connect(self):
self.db = pymysql.Connect(host=self.host, user=self.user, passwd=self.pwd, db=self.name)
self.cursor = self.db.cursor()
def close(self):
self.cursor.close()
self.db.close()
def get_one(self, sql):
try:
self.connect()
self.cursor.execute(sql)
res = self.cursor.fetchone()
self.data = res
self.last_sql = sql
self.close()
except:
print("fail to select")
return self
def get_all(self, sql):
try:
self.connect()
self.cursor.execute(sql)
res = self.cursor.fetchall()
self.last_sql = sql
self.data = res
self.close()
except:
print("fail to select")
return self
def get_all_obj(self, sql):
resList = []
fieldList = []
arr = sql.lower().split(" ")
while "" in arr:
arr.remove("")
tableName = arr[arr.index("from")+1]
fieldSql = "select COLUMN_NAME from information_schema.COLUMNS where table_name='%s' and table_schema='%s'" % (
tableName, self.name
)
fields = self.get_all(fieldSql).data
for item in fields:
fieldList.append(item[0])
res = self.get_all(sql).data
for item in res:
obj = {}
count = 0
for x in item:
obj[fieldList[count]] = x
count += 1
resList.append(obj)
self.data = resList
return self
def insert(self, sql):
return self.execute(sql)
def update(self, sql):
return self.execute(sql)
def delete(self, sql):
return self.execute(sql)
def execute(self, sql):
count = 0
try:
self.connect()
count = self.cursor.execute(sql)
self.db.commit()
self.data = self.db.rowcount()
self.last_sql = sql
self.close()
except:
print("fail execute sql")
self.db.rollback()
return count
server.py,代码:
from tornado import ioloop
from tornado import web
from tornado.options import define, options, parse_command_line
import os
mysql = {
"host": "127.0.0.1",
"user": "root",
"pwd": "123",
"db": "mofang"
}
from mysql import MySQL
settings = {
'debug': True,
# 静态文件保存路径
"static_path": os.path.join(os.path.dirname(__file__), 'static'),
# 静态文件url地址前缀
"static_url_prefix":"/static/", # 必须前后有斜杠
"template_path": os.path.join(os.path.dirname(__file__), 'templates'),
"login_url": "/login",
}
define("port", default=8888, type=int, help="设置监听端口号,默认为8888")
class User(web.RequestHandler):
def prepare(self) -> None:
self.db = MySQL(mysql["host"], mysql["user"], mysql["pwd"], mysql["db"])
def get(self):
sql = "SELECT * FROM mf_user"
ret = self.db.get_all_obj(sql).data
self.write("helle,get")
# 设置路由列表
from tornado.web import url
urls = [
(r"/", User),
]
if __name__ == "__main__":
# 创建应用实例对象
parse_command_line()
app = web.Application(urls, **settings)
# 设置监听的端口和地址
app.listen(options.port)
# ioloop,全局的tornado事件循环,是服务器的引擎核心,start表示创建IO事件循环
ioloop.IOLoop.current().start()