Tornado 框架的使用
Tornado
tornado是一个轻量级python的web框架,他是非阻塞式的,而且速度非常快.得利于其 非阻塞的方式和对 epoll 的运用,Tornado 每秒可以处理数以千计的连接,这意味着对于实时 Web 服务来说,Tornado 是一个理想的 Web 框架。
1.最根本的使用
#!/usr/bin/env python
# -*- coding:utf-8 -*-
import tornado.ioloop
import tornado.web
class MainHandler(tornado.web.RequestHandler):
def get(self):
self.write("Hello, world")
application = tornado.web.Application([
(r"/index", MainHandler),
])
if __name__ == "__main__":
application.listen(8888)
tornado.ioloop.IOLoop.instance().start()
2.路由系统
tornado的路由系统其实就是 url 和 类 的对应关系,这里不同于其他框架,其他很多框架均是 url 对应 函数,Tornado中每个url对应的是一个类。
application = tornado.web.Application([
(r"/index", MainHandler),
(r"/story/([0-9]+)", StoryHandler),
])
application.add_handlers('news.baidu.com$', [
(r'/index',NewsHandler),
])
Tornado中原生支持二级域名的路由
3.模版引擎
Tornado 的模板支持“控制语句”和“表达语句”,控制语句是使用 {% 和 %} 包起来的
例如 {% if len(items) > 2 %}。
表达语句是使用 {{ 和 }} 包起来的,
例如 {{ items[0] }}。
Tornado控制语句和对应的 Python 语句的格式基本完全相同。我们支持 if、for、while 和 try,这些语句逻辑结束的位置需要用 {% end %} 做标记。还通过 extends 和 block 语句实现了模板继承。
在使用模板前需要在setting中设置模板路径:"template_path" : "tpl"
例如:
逻辑语句
<ul>
{% for item in list_info %}
<li>{{item}}</li>
{% end %}
</ul>
-----------------------------------------
继承母版
{% block RenderBody %}{% end %}
{% block RenderBody %}
<h1>Index</h1>
在这里写内容
{% end %}
-----------------------------------------
导入固定的小组件
{% include 'header.html' %}
在模板中默认提供了一些函数、字段、类以供模板使用:
escape: tornado.escape.xhtml_escape 的別名
xhtml_escape: tornado.escape.xhtml_escape 的別名
url_escape: tornado.escape.url_escape 的別名
json_encode: tornado.escape.json_encode 的別名
squeeze: tornado.escape.squeeze 的別名
linkify: tornado.escape.linkify 的別名
datetime: Python 的 datetime 模组
handler: 当前的 RequestHandler 对象
request: handler.request 的別名
current_user: handler.current_user 的別名
locale: handler.locale 的別名
_: handler.locale.translate 的別名
static_url: for handler.static_url 的別名
xsrf_form_html: handler.xsrf_form_html 的別名
当然了,还可以自定义自己的功能
4.自定义UIMethod以UIModule
-
UIMethod:方便小巧,适合简单的方法
# uimethods.py def tab(self): return 'UIMethod' ------------------------------ 导入和注册 import uimethods as mt settings = { ......, 'ui_methods': mt, } ------------------------------ 使用就很方便了 {{ tab() }} #记住要加()
-
UIModule:模块化,适合组合模块
from tornado.web import UIModule from tornado import escape class custom(UIModule): def render(self, *args, **kwargs): '''直接返回内容''' def css_files(self): '''返回一个引入式css路径''' def embedded_css(self): '''以字符串的方式返回一个嵌入式的css''' def javascript_files(self): '''返回一个引入式JS路径''' def embedded_javascript(self): '''以字符串的方式返回css语句''' ------------------------------ 导入和注册 import uimodules as md settings = { ......, 'ui_modules': md, } ------------------------------ 使用 {% module custom(123) %} # 可以传参
5.静态文件
对于静态文件,可以配置静态文件的目录和前段使用时的前缀,并且Tornaodo还支持静态文件缓存。
在设置中写上:
settings = {
'template_path': 'template',
'''模版路径'''
'static_path': 'static',
'''实际静态文件路径'''
'static_url_prefix': '/static/',
'''静态文件的默认识别前缀,意思是静态文件的路径不管是什么,这要这里写了/static/,
前端不管真实路径,直接用/static/,tornado会自动查找真实路径'''
}
--------------------------------------------
前端使用
<link href="/static/example.css" rel="stylesheet" />
为避免出现前缀修改后前端全部都要修改的情况,还可以用一个方法,使前端可以自动生成路径的前缀
<link href="{{ static_url("example.css") }}" rel="stylesheet" />
6.Cookie
Tornado中可以对cookie进行操作,并且还可以对cookie进行签名以放置伪造。
self.set_cookie("mycookie", "myvalue")
'''设置cookie'''
self.set_secure_cookie("mycookie", "myvalue")
'''设置加密cookie'''
设置cookie的时候可以加时间戳参数expires=...,意为到这个时间cookie失效
或者expires_day=3,则过期时间是3天,同时设置的话expires优先级更高
注意:设置加密cookie的时候,需在setting中设置加密签名
settings = {
......,
'cookie_secret':'自己填',
}
--------------------------------------------
self.get_cookie('mycookie')
'''获取cookie'''
self.get_secure_cookie('mycookie')
'''获取加密cookie'''
签名Cookie的本质是:
写cookie过程:
将值进行base64加密
对除值以外的内容进行签名,哈希算法(无法逆向解析)
拼接 签名 + 加密值
读cookie过程:
读取 签名 + 加密值
对签名进行验证
base64解密,获取值内容
7.XSRF
Tornado中的跨站请求伪造和Django中的相似.
设置:
settings = {
...
"xsrf_cookies": True,
}
使用:
<form action="/new_message" method="post">
{{ xsrf_form_html() }}
<input type="text"/>
</form>
Ajax使用就在本地获取cookie,携带cookie发送请求
_xsrf = getCookie("_xsrf")
8.异步非阻塞
装饰器 + Future 从而实现Tornado的异步非阻塞
class AsyncHandler(tornado.web.RequestHandler):
@gen.coroutine
def get(self):
future = Future()
future.add_done_callback(self.doing)
yield future
# 或
# tornado.ioloop.IOLoop.current().add_future(future,self.doing)
# yield future
def doing(self,*args, **kwargs):
self.write('async')
self.finish()
当发送GET请求时,由于方法被@gen.coroutine装饰且yield 一个 Future对象,那么Tornado会等待,等待用户向future对象中放置数据或者发送信号,如果获取到数据或信号之后,就开始执行doing方法。
异步非阻塞体现在当在Tornaod等待用户向future对象中放置数据时,还可以处理其他请求。
注意:在等待用户向future对象中放置数据或信号时,此连接是不断开的。
其他
tornado中的请求的一切都可以在self.request中找到.
待定