前言

当你觉得你过得很舒服的时候,你肯定没有在进步。所以我想学习新的东西,然后选择了Tornado。因为我觉得Tornado更匹配目前的我的综合素质。

Tornado学习笔记系列主要参考《introduction to Tornado》一书,网上有中文版,地址为

http://demo.pythoner.com/itt2zh/index.html

当然也参考了大量博客,在此鸣谢!

本系列不适合完全的0基础小白。

简介

Tornado全称Tornado Web Server,是一个用Python语言写成的Web服务器兼Web应用框架,由FriendFeed公司在自己的网站FriendFeed中使用,被Facebook收购以后框架在2009年9月以开源软件形式开放给大众。

Tornado与其他Web框架的区别

以Django为代表的python web应用部署时采用wsgi协议与服务器对接(被服务器托管),而这类服务器通常都是基于多线程的,也就是说每一个网络请求服务器都会有一个对应的线程来用web应用(如Django)进行处理。

考虑两类应用场景

  1. 用户量大,高并发

    如秒杀抢购、双十一某宝购物、春节抢火车票

  2. 大量的HTTP持久连接

    使用同一个TCP连接来发送和接收多个HTTP请求/应答,而不是为每一个新的请求/应答打开新的连接的方法。

    对于HTTP 1.0,可以在请求的包头(Header)中添加Connection: Keep-Alive。

    对于HTTP 1.1,所有的连接默认都是持久连接。

对于这两种场景,通常基于多线程的服务器很难应对。

对于前面提出的这种高并发问题,我们通常用C10K这一概念来描述。C10K—— Concurrently handling ten thousandconnections,即并发10000个连接。对于单台服务器而言,根本无法承担,而采用多台服务器分布式又意味着高昂的成本。如何解决C10K问题?

Tornado在设计之初就考虑到了性能因素,旨在解决C10K问题,这样的设计使得其成为一个拥有非常高性能的解决方案(服务器与框架的集合体)。

Hello Word

第一个py

安装Tornado部分略过,我们直接进入正题,PyCharm没有新建Tornado项目的选项,我们直接新建一个py即可。

 # -*- coding=utf-8 -*-
import tornado.web
import tornado.ioloop class Index(tornado.web.RequestHandler):
# 封装一个类
def get(self):
# get请求进入该方法
# 返回字符串
self.write('Hello World') class Home(tornado.web.RequestHandler):
def get(self):
self.write('Home') if __name__ == '__main__':
app = tornado.web.Application([
# 相当于路由
(r'/', Index),
(r'/home', Home),
])
# 指定端口
app.listen(8000)
# 开启
tornado.ioloop.IOLoop.current().start()

我们直接启动即可,然后使用postman或者浏览器访问(我习惯使用postman)

http://localhost:8000

我们测试一下结果

当服务器收到请求时会进入Application,进入路由顺序查找匹配。匹配到进入相关class,再根据方法进行处理

如果没有对应class报404,没有相应的方法报405

开启多进程

之前说过Tornado的特点便是多进程,但是上面的代码是单进程的,我们需要修改代码来开启多进程

# -*- coding=utf-8 -*-
import tornado.web
import tornado.httpserver
import tornado.ioloop class Index(tornado.web.RequestHandler):
# 封装一个类
def get(self):
# get请求进入该方法
# 返回字符串
self.write('Hello World') class Home(tornado.web.RequestHandler):
def get(self):
self.write('Home') if __name__ == '__main__':
app = tornado.web.Application([
# 相当于路由
(r'/', Index),
(r'/home', Home),
])
# 手动生成server
http_server = tornado.httpserver.HTTPServer(app)
# 指定端口
http_server.bind(8000)
# 开启多进程
http_server.start(0)
# 开启
tornado.ioloop.IOLoop.current().start()

注意:指定多进程在linux上可行,在windows会报错,因为fork这个系统命令,只在linux中才有用。所以windows请留空(默认为1)或者填1

http_server.start(num_processes=1)方法指定开启几个进程,参数num_processes默认值为1,即默认仅开启一个进程;如果num_processes为None或者<=0,则自动根据机器硬件的cpu核芯数创建同等数目的子进程;如果num_processes>0,则创建num_processes个子进程。

虽然tornado给我们提供了一次开启多个进程的方法,但是由于:

  • 每个子进程都会从父进程中复制一份IOLoop实例,如过在创建子进程前我们的代码动了IOLoop实例,那么会影响到每一个子进程,势必会干扰到子进程IOLoop的工作;
  • 所有进程是由一个命令一次开启的,也就无法做到在不停服务的情况下更新代码;
  • 所有进程共享同一个端口,想要分别单独监控每一个进程就很困难。

不建议使用这种多进程的方式,而是手动开启多个进程,并且绑定不同的端口。

Tornado options组件(命令行加参数)

像端口这种易改变的配置写在代码里则会有解耦性的问题,这个时候我们就需要Tornado的options组件。他可以帮助我们实现全局参数的定义存储和转换。

tornado.options.define()

参数有

  • name 选项变量名,须保证全局唯一性,否则会报“Option 'xxx' already defined in ...”的错误;
  • default 选项变量的默认值,如不传默认为None;
  • type 选项变量的类型,从命令行或配置文件导入参数的时候tornado会根据这个类型转换输入的值,转换不成功时会报错,可以是str、float、int、datetime、timedelta中的某个,若未设置则根据default的值自动推断,若default也未设置,那么不再进行转换。可以通过利用设置type类型字段来过滤不正确的输入。
  • multiple 选项变量的值是否可以为多个,布尔类型,默认值为False,如果multiple为True,那么设置选项变量时值与值之间用英文逗号分隔,而选项变量则是一个list列表(若默认值和输入均未设置,则为空列表[])。
  • help 选项变量的帮助提示信息,在命令行启动tornado时,通过加入命令行参数 --help 可以查看所有选项变量的信息(注意,代码中需要加入tornado.options.parse_command_line())。
define("port", default=8000, help="run on the given port", type=int)

此处做到可以在启动时输入一个port,来指定端口,不传默认为8000,输入类型为int

python tdo_helloword.py --port=9000

指定端口为9000

那么代码修改为

# -*- coding=utf-8 -*-
import tornado.web
import tornado.ioloop
import tornado.httpserver
import tornado.options
# 可以多行
tornado.options.define("port", default=8000, help="run on the given port", type=int) class Index(tornado.web.RequestHandler):
# 封装一个类
def get(self):
# get请求进入该方法
# 返回字符串
self.write('Hello World') class Home(tornado.web.RequestHandler):
def get(self):
self.write('Home') if __name__ == '__main__':
# 注意加上这句
tornado.options.parse_command_line()
app = tornado.web.Application([
# 相当于路由
(r'/', Index),
(r'/home', Home),
])
# 手动生成server
http_server = tornado.httpserver.HTTPServer(app)
# 指定端口
http_server.bind(tornado.options.options.port)
# 开启多进程
http_server.start(1)
# 开启
tornado.ioloop.IOLoop.current().start()

我们在命令行加入port参数

python tdo_helloword.py --port=9999

查看效果

Tornado options组件(从配置文件导入)

配置文件格式要对

我们要更新一下代码

# -*- coding=utf-8 -*-
import tornado.web
import tornado.ioloop
import tornado.httpserver
import tornado.options
# 可以多行
tornado.options.define("port", default=8000, help="run on the given port", type=int) class Index(tornado.web.RequestHandler):
# 封装一个类
def get(self):
# get请求进入该方法
# 返回字符串
self.write('Hello World') class Home(tornado.web.RequestHandler):
def get(self):
self.write('Home') if __name__ == '__main__':
# 注意加上这句
# tornado.options.parse_command_line()
# 从文件读取配置
tornado.options.parse_config_file('./config')
app = tornado.web.Application([
# 相当于路由
(r'/', Index),
(r'/home', Home),
])
# 手动生成server
http_server = tornado.httpserver.HTTPServer(app)
# 指定端口
http_server.bind(tornado.options.options.port)
# 开启多进程
http_server.start(1)
# 开启
tornado.ioloop.IOLoop.current().start()

这样便是读取配置文件来启动tornado

关闭日志

在我们访问网站的时候,我们会发现屏幕打印了访问和返回的记录,我们可以将他关闭

想要关闭,我们可以在开启时加上--logging=none

python td_helloword.py --logging=none

或者修改代码为

 # -*- coding=utf-8 -*-
import tornado.web
import tornado.ioloop
import tornado.httpserver
import tornado.options
# 可以多行
tornado.options.define("port", default=8000, help="run on the given port", type=int) class Index(tornado.web.RequestHandler):
# 封装一个类
def get(self):
# get请求进入该方法
# 返回字符串
self.write('Hello World') class Home(tornado.web.RequestHandler):
def get(self):
self.write('Home') if __name__ == '__main__':
# 不打印日志
tornado.options.options.logging = None
tornado.options.parse_command_line()
# 从文件读取配置
# tornado.options.parse_config_file('./config')
app = tornado.web.Application([
# 相当于路由
(r'/', Index),
(r'/home', Home),
])
# 手动生成server
http_server = tornado.httpserver.HTTPServer(app)
# 指定端口
http_server.bind(tornado.options.options.port)
# 开启多进程
http_server.start(1)
# 开启
tornado.ioloop.IOLoop.current().start()

Tornado学习笔记(一) helloword/多进程/启动参数的更多相关文章

  1. (转)redis 学习笔记(1)-编译、启动、停止

    redis 学习笔记(1)-编译.启动.停止   一.下载.编译 redis是以源码方式发行的,先下载源码,然后在linux下编译 1.1 http://www.redis.io/download 先 ...

  2. Windows phone 8 学习笔记(4) 应用的启动

    原文:Windows phone 8 学习笔记(4) 应用的启动 Windows phone 8 的应用除了可以直接从开始菜单以及应用列表中打开外,还可以通过其他的方式打开.照片中心.音乐+视频中心提 ...

  3. tornado 学习笔记1 引言

    从事软件开发这行业也快5年啦,其实从事的工作也不完全是软件开发,软件开发只是我工作中的一部分.其中包括课题研究.信息化方案设计.软件开发.信息系统监理.项目管理等工作,比较杂乱.开发的软件比较多,但是 ...

  4. Android学习笔记1 android adb启动失败问题 adb server is out of date. killing...

    下面是Android的学习笔记,原文地址. 我是使用adb devices出现如下红字错误, 使用第一种方法方法,结果关掉豌豆荚就可以了. android adb启动失败问题 adb server i ...

  5. FAT文件系统规范v1.03学习笔记---1.保留区之启动扇区与BPB

    1.前言 本文主要是对Microsoft Extensible Firmware Initiative FAT32 File System Specification中文翻译版的学习笔记. 每个FAT ...

  6. 学习笔记之Struts2—浅析接收参数

    最近自己通过视频与相关书籍的学习,对action里面接收参数做一些总结与自己的理解. 0.0.接收参数的(主要)方法   使用Action的属性接收参数 使用DomainModel接收参数 使用Mod ...

  7. Python学习笔记之selenium 定制启动 chrome 的选项

    在自动化中,默认情况下我们打开的就是一个普通的纯净的chrome浏览器,而我们平时在使用浏览器时,经常就添加一些插件,扩展,代理之类的应用.所以使用 selenium 时,我们可能需要对 chrome ...

  8. MySQL学习笔记——MySQL5.7的启动过程(一)

    MySQL的启动函数在 sql/main.cc 文件中. main.cc: extern int mysqld_main(int argc, char **argv); int main(int ar ...

  9. Tornado学习笔记12 tornado.httpserver-.非阻塞的Http服务器

    是一个非阻塞的,单线程的Http 服务器. 一般地,应用程序很少与HttpServer类直接交互,除非在进程开始时启动服务时(甚至在使用tornado.web.Applicaiton.listen时也 ...

随机推荐

  1. js 数组拷贝与深拷贝

    1.对于普通数组(数组元素为数字或者字符串) var _testCopy = [1,2,3].concat();//拷贝数组(浅拷贝) 2.对于对象数组 (深拷贝) //形如var _objArr=[ ...

  2. NPOI导出Excel2007-xlsx

    今天在用npoi导出xls时会报错,经过在网上查找资料,找到一篇博客文章介绍的,原文地址https://www.cnblogs.com/spring_wang/p/3160020.html 1.今天再 ...

  3. orcle查看表空间数据文件使用情况

    -- 查看表空间数据文件使用情况select a.*, round(a.usedgb/a.maxgb*100) || '%' usedPer from (select t.TABLESPACE_NAM ...

  4. 用PHP来获取access_token

    $url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=".A ...

  5. 第26月第28天 avplayer cache

    1.urlsession https - (void)URLSession:(NSURLSession *)session didReceiveChallenge:(NSURLAuthenticati ...

  6. Spark Streaming连接Kafka的两种方式 direct 跟receiver 方式接收数据的区别

    Receiver是使用Kafka的高层次Consumer API来实现的. Receiver从Kafka中获取的数据都是存储在Spark Executor的内存中的,然后Spark Streaming ...

  7. js判断浏览器类型和版本

    原网址:http://www.cnblogs.com/rubylouvre/archive/2009/10/14/1583362.html 除了另无它法,肯定不使用navigator.userAgen ...

  8. svn 的truck、tag、 merge

    参考文章 :    https://blog.csdn.net/keda8997110/article/details/21813035

  9. GDI+学习---2.GDI+编程模式及组成类

    在使用GDI+的时候,您不必像在GDI中那样关心设备场景句柄,只需简单地创建一个Graphics对象,然后以您熟悉的面向对象的方式(如myGraphicsObject.DrawLine(paramet ...

  10. python - 文件系统和文件

    文件系统和文件        文件系统是os用于明确磁盘或分区上的文件的方法和数据结构--即在磁盘上组织文件的方法        计算机文件,是存储在某种长期储存设备或临时存储设备中的一段数据流,并且 ...