在之前的其他博客中介绍了Django,这里介绍一下Tornado。两者的区别可以总结为:

  1. django大而全,适合小型的压力不大的项目,一旦压力上来其是扛不住的,毕竟一是太重,而是非异步。 但是好处就是什么都有,你能想到的功能其都有对应contrib组件给你用
  2. tornado好处是epoll异步性能高,还支持长连接。 坏处是:功能/第三方库相对少,而且想多实例还要自己去配置,再者没有很成熟的配套framework,而只是提供了核心的功能,其余的都需要你自己来做。

1.tornado初见

使用Tornado之前需要pip安装

pip3 install tornado

参考教程,tornado书写如下app.py脚本

import tornado.ioloop
from tornado.web import RequestHandler, Application
from tornado.httpserver import HTTPServer
from tornado.options import options, definedefine('port', default=8000, help='监听端口')class HelloHandler(RequestHandler):def get(self):self.write('hello world')if __name__ == '__main__':options.parse_command_line()handlers_routes = [(r'/', HelloHandler)]app = Application(handlers=handlers_routes)http_server = HTTPServer(app)http_server.listen(options.port)tornado.ioloop.IOLoop.current().start()

运行python app.py,在浏览器中输入http://127.0.0.1:8000。展示结果如下

这一点来看和Django比较相似,很“容易”展示出需要的首个web请求界面

Tornado中最重要的一个模块是web, 它就是包含了 Tornado 的大部分主要功能的 Web 框架。其它的模块都是工具性质的, 以便让 web 模块更加有用 后面的 Tornado 攻略 详细讲解了 web 模块的使用方法。Tornado中主要有以下模块信息:
主要模块

  • web - FriendFeed 使用的基础 Web 框架,包含了 Tornado 的大多数重要的功能
  • escape - XHTML,JSON, URL 的编码/解码方法
  • database - 对 MySQLdb 的简单封装,使其更容易使用
  • template - 基于Python 的 web 模板系统
  • httpclient - 非阻塞式 HTTP 客户端,它被设计用来和 web 及 httpserver协同工作
  • auth - 第三方认证的实现(包括 Google OpenID/OAuth、Facebook Platform、YahooBBAuth、FriendFeed OpenID/OAuth、Twitter OAuth)
  • locale - 针对本地化和翻译的支持
  • options - 命令行和配置文件解析工具,针对服务器环境做了优化

底层模块

  • httpserver - 服务于 web 模块的一个非常简单的 HTTP 服务器的实现
  • iostream - 对非阻塞式的 socket 的简单封装,以方便常用读写操作
  • ioloop - 核心的 I/O 循环

2.释义

2.1.RequestHandler

RequestHandler 是专门用来处理客户端请求的类,需要自定义一个类并继承它并实现对应的方法。我在示例里只实现了get方法,用它来处理get请求,如果你需要处理post请求就需要实现post方法。在RequestHandler类,一共定义了7个处理请求的方法

def _unimplemented_method(self, *args: str, **kwargs: str) -> None:raise HTTPError(405)head = _unimplemented_method  # type: Callable[..., Optional[Awaitable[None]]]
get = _unimplemented_method  # type: Callable[..., Optional[Awaitable[None]]]
post = _unimplemented_method  # type: Callable[..., Optional[Awaitable[None]]]
delete = _unimplemented_method  # type: Callable[..., Optional[Awaitable[None]]]
patch = _unimplemented_method  # type: Callable[..., Optional[Awaitable[None]]]
put = _unimplemented_method  # type: Callable[..., Optional[Awaitable[None]]]
options = _unimplemented_method  # type: Callable[..., Optional[Awaitable[None]]]

Tornado 的 Web 程序会将 URL 或者 URL 范式映射到 tornado.web.RequestHandler 的子类上去。在其子类中定义了 get() 或 post() 等如上的各种方法,用以处理不同的 HTTP 请求。

2.2.Application

Application对应整个web应用,在创建Application时需要指定handlers参数,它是一个列表,列表里的元素是元组,元组的第一个元素是正则表达式,用来匹配请求的path,元组的第二个元素是自定义的Handler, 这个Handler用来处理与正则表达式相匹配的请求,因此Application的功能是用来匹配请求路由映射

handlers参数解决的是请求与处理请求类之间的映射关系,它的作用等同于flask的路由。

2.3.HTTPServer

tornado 既是一个web框架也是一个web服务器,web框架让我们专注于处理业务逻辑,Application是web框架体现,HTTPServer是web服务器的体现。

怎么区分他们呢?你就看在哪里设置监听host和端口号,host与端口号是一组和服务器相关的概念,web框架不涉及这些东西,web框架只帮助你快速的进行web应用的开发,而host和端口号是部署的时候用的。http_server.listen(8000) 表明要监听8000端口,listen方法还可以设置address参数,默认是空字符串,表示监听地址是0.0.0.0。

def listen(self, port: int, address: str = "") -> None:

引入了tornado.httpserver模块,顾名思义,它就是tornado的HTTP服务器实现。

这里创建了一个HTTP服务器实例http_server,因为服务器要服务于我们刚刚建立的web应用,将接收到的客户端请求通过web应用中的路由映射表引导到对应的handler中,所以在构建http_server对象的时候需要传出web应用对象app。http_server.listen(8000)将服务器绑定到8000端口

2.4 ioloop

ioloop模块是tornado的精髓所在,tornado之所以具有较高的性能,全赖ioloop提供的异步io能力。对于ioloop,在初学阶段,你只需要掌握这一行代码就足够了

tornado.ioloop.IOLoop.current().start()

2.5 define 与 options

define和options相互配合以实现tornado web应用的配置,define定义web启动时的各项参数,options.parse_command_line()函数解析启动命令中的参数,port参数设置有默认值,如果我在启动时不指定端口,则使用默认端口号8000,我也可以指定端口号

python web.py --port=8001

程序启动后,options.port的值是8001

3.路由模块

所谓路由模块是指域名后的资源定位符,比如get请求为域名+查询字符串串。tornado中路由系统建立请求path和处理该类请求代码(函数,类)之间的映射关系。

3.1.路由系统示例

import tornado.ioloop
from tornado.web import RequestHandler, Application
from tornado.httpserver import HTTPServer
from tornado.options import options, definedefine('port', default=8000, help='监听端口')class HelloHandler(RequestHandler):def get(self):self.write('hello world')class IndexHandler(RequestHandler):def get(self):self.write('welcome to IndexHandler')if __name__ == '__main__':options.parse_command_line()handlers_routes = [(r'/', HelloHandler),(r'/index', IndexHandler)]app = Application(handlers=handlers_routes)http_server = HTTPServer(app)http_server.listen(options.port)tornado.ioloop.IOLoop.current().start()

运行之后,在浏览器中输入 http://127.0.0.1:8000/index 结果如下所示:

在创建Application对象时,需要设置handlers参数,如上代码所示,handlers_routes 是一个列表,列表里的元素是元组,元素第一个元素是正则表达式,描述的是请求的path,第二个元素是RequestHandler类,处理path符合前面正则表达式的请求。

3.2.动态路由

动态路由比较典型的例子就是实用Django访问一个blog的示例。这个帖子比较常见。tornado 也支持动态路由,结合正则表达式分组的知识

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# This module is ***
# @Time : 2022/10/22 21:59
# @Author : renhuaxi
# @Site :
# funciton:
# @File : pathdemo.py
import tornado.ioloop
from tornado.web import RequestHandler, Application
from tornado.httpserver import HTTPServer
from tornado.options import options, definedefine('port', default=8000, help='监听端口')class HelloHandler(RequestHandler):def get(self, name):self.write('hello world')class IndexHandler(RequestHandler):def get(self):self.write('welcome to IndexHandler')# 新增用户指定的路由请求
class UserHandler(RequestHandler):def get(self, name):self.write(f'welcome {name}')if __name__ == '__main__':options.parse_command_line()handlers_routes = [(r'/', HelloHandler),(r'/index', IndexHandler),# 新增一个路由转发关系(r'/user/(.*)', UserHandler)]app = Application(handlers=handlers_routes)http_server = HTTPServer(app)http_server.listen(options.port)tornado.ioloop.IOLoop.current().start()

输出结果如下所示:

r’/user/(.)’ 是一个正则表达式,(.) 是一个分组,其匹配到的内容将传递给UserHandler类的get方法中的name参数。正则表达式里有几个分组,在对应的处理请求的方法里就要有几个参数。

下面的正则表达式也成立,而且代码更利于阅读,你可以根据正则表达式里的捕获分组准确的理解path里每一个分组的含义

(r'/user/(?P<name>.*)', UserHandler)

3.3.URLSpec

from tornado.routing import URLSpechandlers_routes = [(r'/', HelloHandler),URLSpec(r'/index', IndexHandler),(r'/user/(?P<name>.*)', UserHandler)]

可以使用URLSpec类来确定路径和处理请求类之间的映射关系,这与使用元组在效果上是相同的。使用元组,tornado会根据元组里的数据生成Rule对象,而URLSpec是Rule的子类

3.4.add_handlers

Application 类的add_handlers方法允许你自由添加路由规则,不仅如此,还可以向指定的host进行添加。

app = Application(handlers=handlers_routes)app.add_handlers(r'.*', [(r'/user/(?P<name>.*)', UserHandler)])

add_handlers的第一个参数是host_pattern,根据host_pattern生成Matcher对象,只有请求头里的host匹配了host_pattern才会生效,r’.’ 表示任意字符重复0次或多次,任意host都可以使用这些路由规则。app = Application(handlers=handlers_routes) 这种写法在源码里生成的是AnyMatches对象,同r’.'一样,匹配任意host。

如果你的web服务有多个域名,而且希望不同的域名有不同的路由规则,那么你可以使用add_handlers方法来实现,在/etc/hosts里配置如下域名解析

  • 127.0.0.1 mycool.com
  • 127.0.0.1 mypythonweb.com
    修改web应用如下所示
app = Application(handlers=handlers_routes)app.add_handlers(r'mycool.com', [(r'/user/(?P<name>.*)', UserHandler)])

访问http://mycool.com:8000/user/dongsheng 可以得到正确响应,而访问http://mypythonweb.com:8000/user/dongsheng ,得到的就是一个404 not found, host_pattern 可以匹配mycool.com,但不能匹配mypythonweb.com,因此(r’/user/(?P.*)’ 对于mypythonweb.com是不可见的

4.RequestHandler子类处理http请求

使用tornado进行web编程的关键是自定义继承RequestHandler的子类并实现特定的方法,RequestHandler等价于flask框架里的视图,对RequestHandler的理解和使用将决定你能否掌握tornado框架。这是连接请求与web服务的关键环节之一。

4.1.self.request对象

处理客户端的请求,最重要的当然是获的请求的参数,这一点已经在获取请求参数那一篇教程中进行了讲解,除此以外,请求的headers也是十分重要的信息,与请求有关的信息都存储在self.request对象中.

import tornado.ioloop
from tornado.web import RequestHandler, Application
from tornado.httpserver import HTTPServer
from tornado.options import options, definedefine('port', default=8000, help='监听端口')class HelloHandler(RequestHandler):def get(self):print('************输出headers信息************')print(self.request.headers)             # 字典形式存储的headers信息print('************输出host信息************')print(self.request.headers['host'])     # 获取某一个首部print('************输出path信息************')print(self.request.path)                # 请求的pathself.write('ok')if __name__ == '__main__':options.parse_command_line()handlers_routes = [(r'/', HelloHandler)]app = Application(handlers=handlers_routes)http_server = HTTPServer(app)http_server.listen(options.port)tornado.ioloop.IOLoop.current().start()

浏览器输入http:127.0.0.1:8000之后,在控制台查看输出信息

self.request 对象的类型是HTTPServerRequest, 存在于tornado.httputil模块中,这个对象的属性不仅仅包含headers,还包括请求的method,uri, version,body, protocol, remote_ip,与请求有关的所有信息都包含在这个对象里,甚至包括最底层的socket连接,你所需要的信息都可以通过self.reqeust对象获取。

4.2 每一个请求关联一个Requesethandler对象

tornado每收到一个请求都会创建出一个RequestHandler对象对请求进行处理,这一点,可以通过实验来证实,我在处理get请求的方法里输出self对象的内存地址,每次请求到来时,输出的内存地址都是不相同的

# @File : requesthandlerdemo.py
import tornado.ioloop
from tornado.web import RequestHandler, Application
from tornado.httpserver import HTTPServer
from tornado.options import options, definedefine('port', default=8000, help='监听端口')class HelloHandler(RequestHandler):def get(self):print('************输出headers信息************')print(self.request.headers)             # 字典形式存储的headers信息print('************输出host信息************')print(self.request.headers['host'])     # 获取某一个首部print('************输出path信息************')print(self.request.path)                # 请求的pathprint('************输出id信息************')print(id(self))self.write(str(id(self)))if __name__ == '__main__':options.parse_command_line()handlers_routes = [(r'/', HelloHandler)]app = Application(handlers=handlers_routes)http_server = HTTPServer(app)http_server.listen(options.port)tornado.ioloop.IOLoop.current().start()

输出信息如下所示:

4.3.initial 和 prepare

initialize和prepare 是RequestHandler的两个很重要的方法,他们有什么功能作用,又有什么区别呢?

initialize 是框架预留的一个初始化时加载自定义内容的钩子,其会在http请求方法之前调用,prepare 在执行对应的请求方法之前调用。在执行顺序上,先是initialize而后是prepare方法。

initialize 处理从URLSpec接收到的参数,你可以在创建app时对initialize所能接收的参数进行设置,除此以外,你不能在initialize中做更多的事情,比如调用write, finish方法。

而在prepare里,你可以调用这些方法,假设你要写一个根据IP进行过滤的逻辑,那么你应该写在prepare方法里。在执行get或post之前在prepare方法里对客户端的IP进行过滤,如果发现IP不符合要求,则可以调用finish方法结束请求。注意,不要使用write方法,write方法在这里不会结束请求,你在这里write的内容会和后续处理请求的方法(get,或post)所返回的内容连接在一起返回。

下面的示例中,我在prepare方法里结束请求

# @File : initialAndPrepareDemo.py
import tornado
import tornado.ioloop
from tornado.web import RequestHandler
from tornado.web import URLSpecclass HelloHandler(RequestHandler):def initialize(self, db):print('initialize')self.db = db#  self.finish('over')  这里不可以调用finishdef prepare(self):print('prepare')self.finish('over')def get(self):self.write(f'{self.db} ok')url_handlers = [URLSpec(r'/hello', HelloHandler, {'db': 'mysql'})
]app = tornado.web.Application(url_handlers)
http_server = tornado.httpserver.HTTPServer(app)
http_server.listen(8990)
http_server.start()tornado.ioloop.IOLoop.instance().start()

4.4.write与finish的区别

在第三节的例子里,处理请求的get方法使用write方法返回响应数据,而prepare方法里使用finish返回数据,这两个方法的区别是什么呢?

write和finish都可以向客户端返回数据,不同之处在于finish执行时,返回动作终结,如果你还有一些逻辑且这些逻辑和返回数据无关,那么你可以放在finish后面执行,对于客户端来说,它已经收到响应结果,你放在finish后面的逻辑所占用的时间与客户端无关。

write也是向客户端返回数据,但只有在遇到finish或者return后才会真正的向前端返回数据。因此,在处理请求的方法结束以前,你可以多次调用write方法,而finish则不可以。

Tornado初见(一)相关推荐

  1. python tornado教程_Tornado 简单入门教程(零)——准备工作

    前言: 这两天在学着用Python + Tornado +MongoDB来做Web开发(哈哈哈这个词好高端).学的过程中查阅了无数资料,也收获了一些经验,所以希望总结出一份简易入门教程供初学者参考.完 ...

  2. python中tornado的第一个例子

    python中tornado的第一个例子 1  先安装tornado pip install tornado 2 新建tor.py 记住不能建立 tornado.py 这样的名字  不然会报错 Imp ...

  3. tornado+nginx上传视频文件

    [http://arloz.me/tornado/2014/06/27/uploadvideotornado.html] [NGINX REFRER: Nginx upload module] 由于t ...

  4. tornado源码分析

    tornado源码分析 本源码为tornado1.0版本 源码附带例子helloworld import tornado.httpserver import tornado.ioloop import ...

  5. Tornado自定义分布式session框架

    Tornado自定义分布式session框架 一.session框架处理请求执行的流程: 1.服务器端生成随机的cookie字符串 2.浏览器发送请求,服务器将cookie返回给浏览器. 3.服务器在 ...

  6. tornado 入门

    Overview FriendFeed是一款使用 Python 编写的,相对简单的 非阻塞式 Web 服务器.其应用程序使用的 Web 框架看起来有些像 web.py 或者 Google 的 weba ...

  7. AOP之PostSharp初见-OnExceptionAspect

    PostSharp 这个静态植入的aop框架我就不多说了,在以前的aop文件,我们也尝试用MSBuild+Mono.Cicel理解静态植入AOP的原理.最近公司准备购买Postsharp做一些AOP, ...

  8. python web项目案例教程_Python Web开发案例教程(慕课版)——使用Flask、Tornado、Django...

    第1章 Web开发基础1 1.1 Web概述 2 1.1.1 什么是Web 2 1.1.2 Web应用程序的工作原理 2 1.1.3 Web的发展历程 2 1.2 Web前端开发基础 4 1.2.1 ...

  9. sae python连接mysql_SAE Tornado 应用连接并使用 Mysql

    今天因为要提供几个开放的接口给我毕设相关的平台调用,所以又开始折腾之前在SAE上搭的Tornado应用. 之前记录过一个 如何在 SAE 上使用 Tornado,这次续上,关于在SAE里使用Torna ...

最新文章

  1. git clone 多个_如何通过Git参与项目开发
  2. 一张图片解决 Python 所有内置异常
  3. 最大独立匹配_新车|升级柴油国六动力,配后排独立座椅,瑞风M5新车型上市...
  4. VMware vSphere Storage Appliance (VSA) 5.1 群集部署
  5. 从Oracle Database 角度来看浪潮天梭K1主机的操作系统选择
  6. 指标公式c语言源码下载,通达信99.75%成功率指标公式 源码
  7. C语言程序设计C语言之父,C程序设计语言-美-里奇-C语言之父-机械工业出版社
  8. Excel从入门到精通--基础篇
  9. 如何向别人推荐(分享)一本书
  10. 百度官方的6个SEO建议
  11. 计算机word文本段落位置互换,word中调换位置 用word怎么使两个段落互换位置
  12. AQS的前菜—详解CLH队列锁
  13. Maven-仓库概念,下载与配置
  14. MT6737芯片技术资料集锦下载
  15. 什么样的打码网站算正规的打码网站
  16. SUMO使用E2检测器获取信号交叉口车道信息和信号控制方案
  17. 阿诺德·施瓦辛格的演讲(拼命才会优秀,自律才会自由~)
  18. css鼠标滑过图标显示_CSS和jQuery教程:苹果风格的花式图标滑出导航
  19. java获取字典所有的key_java字典,多层字典,斗地主发牌,实例展示
  20. 计算机象棋思维原理,这些象棋原理只有象棋大师才懂,几分钟就能学会

热门文章

  1. 淘宝流量来源分析,如何做好提升?
  2. html网页设计实训原理,网页设计实习目的及意义
  3. Nero9到底想干什么啊?
  4. 汽车智能座舱中 显示屏市场战略趋势分析 上篇
  5. 亚马逊云科技 AI For Good-2022优秀方案开源分享——OC
  6. 常见问题(持续更新)
  7. #走进图灵#第二章:无名的二战英雄
  8. 100Mbps和100Mb/s有什么不同
  9. 互联网和人类行为的博弈均衡
  10. C++运算符重载中有些方法为什么需要定义为友元函数