トルネード竜巻の研究ノート(c)の要求情報/ステータスコード

簡単な紹介

フルネームトルネードWebサーバーを竜巻、自身のウェブサイトのFriendFeed FriendFeedの中で会社が使用するPythonのWebサーバーとWebアプリケーションフレームワークで書かれている、Facebookのフレームワークによって買収された後には、2009年9月におけるオープンソースソフトウェアの形で一般に公開されています。

他のWebフレームワークとトルネード差

ジャンゴに配備(サーバによってホストされる)プロトコルWSGIアバットメントを使用して、PythonのWebアプリケーションサーバーを表現し、そのような通常のマルチスレッドサーバをベースとする、ネットワーク要求に、すなわち、各サーバは、対応するスレッドとを有します処理のためにウェブアプリケーション(例えば、ジャンゴ)。

シナリオの2種類を考えてみましょう

  1. ユーザーのボリューム、高い同時実行

    このようなスパイクラッシュ、二から一一ショッピング宝物として、春祭りの列車の切符をつかみます

  2. HTTPの永続的な接続の数が多いです

    代わりに、各新規リクエスト/レスポンス開放の接続の新しい方法、HTTPリクエスト/レスポンスの複数の送信および受信するために同じTCP接続。

    キープアライブ:HTTP 1.0のために、接続要求ヘッダ(ヘッダ)を添加してもよいです。

    HTTP 1.1、すべての接続のデフォルトは、永続的な接続です。

通常、マルチスレッドサーバーに基づいたシナリオ、両方の場合に対応することは困難です。

先に述べたこの高並行性の問題のために、私たちはしばしばコンセプトC10Kを記述するために使用します。C10K--は、同時に10 thousandconnections、すなわち万個の同時接続を処理します。順番に分散された複数のサーバーの使用は、高いコストを意味している単一のサーバーの場合は、単純に、余裕はありません。C10Kはどのように問題を解決するには?

トルネードは初期の設計で考慮パフォーマンス要因を取り、問題を解決するために設計されたC10Kは、このデザインは、それは非常に高性能なソリューション(サーバやフレームワークの集まり)となっています。

こんにちはワード

最初PY

トルネードは私たちが疑問に直接移動、設置セクションをスキップし、PyCharmなしの新しいオプショントルネードプロジェクトは、我々が直接新しいPYを作成することができます。

#1 - * - = UTF-コーディング8 - * - 
 2インポートtornado.web 
 3インポートtornado.ioloop。
 4。
 5。
 6クラスインデックス(tornado.web.RequestHandler)。
 。7つの#パッケージ化クラス
 8 DEF GET(自己):
 9#。取得要求は、プロセスに入る
10の返す文字列#
。( 'Hello Worldの')11 self.write 
12は
13です
:14クラスホーム(tornado.web.RequestHandler)
15は、DEFを取得(自己):
16 self.write( 'ホーム')
。17 
18は、
19 __name__ == '__main__'の場合:
20 tornado.web.Applicationアプリ=である([ 
21#経路に相当する
22である(R&LT '/'、インデックス)、
23である(R&LT '/ホーム'、ホーム)、
24]) 
25 #ポートを指定します
26でありますapp.listen(8000) 
27#开启
28 tornado.ioloop.IOLoop.current()。()開始

私たちは直接起動して、郵便配達にアクセスするためにブラウザを使用するか、(私は郵便配達を使用するために使用される)ことができます

http:// localhostを:8000

当社のテスト結果

 

サーバは、要求がアプリケーションを入力します受信した場合、一致するものを見つけるために、ルーティングの順序を入力してください。関連するクラスを入力して一致した後、法に従って処理

404をパケットに対応するクラスが存在しない場合、パケット405には、対応する方法ではありません

マルチプロセスを開きます。

トルネード機能は、マルチプロセスですが、上記のコードは、単一のプロセスである前に、我々はマルチプロセスを開くために、コードを変更する必要があると述べました

# -*- 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

或者修改代码为

 1 # -*- coding=utf-8 -*-
 2 import tornado.web
 3 import tornado.ioloop
 4 import tornado.httpserver
 5 import tornado.options
 6 # 可以多行
 7 tornado.options.define("port", default=8000, help="run on the given port", type=int)
 8 
 9 
10 class Index(tornado.web.RequestHandler):
11     # 封装一个类
12     def get(self):
13         # get请求进入该方法
14         # 返回字符串
15         self.write('Hello World')
16 
17 
18 class Home(tornado.web.RequestHandler):
19     def get(self):
20         self.write('Home')
21 
22 
23 if __name__ == '__main__':
24     # 不打印日志
25     tornado.options.options.logging = None
26     tornado.options.parse_command_line()
27     # 从文件读取配置
28     # tornado.options.parse_config_file('./config')
29     app = tornado.web.Application([
30         # 相当于路由
31         (r'/', Index),
32         (r'/home', Home),
33     ])
34     # 手动生成server
35     http_server = tornado.httpserver.HTTPServer(app)
36     # 指定端口
37     http_server.bind(tornado.options.options.port)
38     # 开启多进程
39     http_server.start(1)
40     # 开启
41     tornado.ioloop.IOLoop.current().start()

路由

路由的匹配

Tornado的路由匹配采用的是正则匹配

一般情况下不需要多复杂的正则,正则的基本规则如下(站长之家)

举个例子

(r'/sum/(\d+)', Sum),

该代码指匹配 /sum/  后跟至少一个数字且只有数字的情况

* 需要注意的是网络上传输都是字符串

类 Sum编写

class Sum(tornado.web.RequestHandler):
    # 数字类
    def get(self, sum):
        # 获取数字并返回
        self.write('%s,%s' % (type(sum), sum))

我们访问 http://localhost:8001/sum/12 时

返回正常

我们访问 http://localhost:8001/sum/1a2  http://localhost:8001/sum/a12  http://localhost:8001/sum/12a  http://localhost:8001/sum/a 时,均会报出404错误,证明没有匹配到路由

 

同理,当我们需要匹配两个参数时

(r'/(\w+)/stuggle/(\d+)', Stugggle),

接收时接收两个参数即可

def get(self, st, ins):
  pass

 post参数

与get一样,post请求会寻找到该视图的 post 方法

我们给视图 Hello 增加一个post

class Hello(tornado.web.RequestHandler):
    # 封装一个类
    def get(self):
        # get请求进入该方法
        self.write('Hello')
    def post(self):
        # post请求
        txt = self.get_argument('txt')
        self.write(txt)

self.get_argument('txt') 指获取post传参中 Key 为 txt 的值,路由无需改动

 get参数

get获取参数与上面的post没有差别

我们修改get方法来进行测试

    def get(self):
        # get请求进入该方法
        arg = self.get_argument('arg')
        arg1 = self.get_argument('arg1')
        self.write('%s+%s' % (arg,arg1))

 

 

但是需要注意的是, get_argument  在获取不到该key时会报错,抛出400错误

 

  get_argument  方法其实可以接收三个参数

get_argument(name,default=_ARG_DEFAULT,strip=True)

第一个参数就是key的值,第二个参数为如果接收不到默认的值,第三个是默认去除前后空格

一般情况下我们第二个参数传 None

    def get(self):
        # get请求进入该方法
        arg = self.get_argument('arg', None)
        arg1 = self.get_argument('arg1', None)
        self.write('%s+%s' % (arg,arg1))
    def post(self):
        # post请求
        txt = self.get_argument('txt', None)
        self.write(txt)

这样就增加了兼容性

Tornado学习笔记(三) 请求方式/状态码

 

本章我们来学习 Tornado 支持的请求方式

请求方式

Tornado支持任何合法的HTTP请求(GETPOSTPUTDELETEHEADOPTIONS)。你可以非常容易地定义上述任一种方法的行为,只需要在 RequestHandler 类中使用同名的方法。(也就是在视图中定义同名的方法)

关于请求方式对应的操作和返回码可以参考 http://www.runoob.com/w3cnote/restful-architecture.html 不过具体还是要与业务进行匹配。

状态码

使用RequestHandler类的set_status()方法显式地设置HTTP状态码。然而,你需要记住在某些情况下,Tornado会自动地设置HTTP状态码。

比如如果访问一个无法匹配的路由,则会返回404报错,而且状态码是404

但是我们在代码中没有指定返回的状态码,常用的有:

  404 Not Found

  Tornado会在HTTP请求的路径无法匹配任何RequestHandler类相对应的模式时返回404(Not Found)响应码。

  400 Bad Request

  如果你调用了一个没有默认值的get_argument函数,并且没有发现给定名称的参数,Tornado将自动返回一个400(Bad Request)响应码。

  405 Method Not Allowed

  如果传入的请求使用了RequestHandler中没有定义的HTTP方法(比如,一个POST请求,但是处理函数中只有定义了get方法),Tornado将返回一个405(Methos Not Allowed)响应码。

  500 Internal Server Error

  当程序遇到任何不能让其退出的错误时,Tornado将返回500(Internal Server Error)响应码。你代码中任何没有捕获的异常也会导致500响应码。

  200 OK

  如果响应成功,并且没有其他返回码被设置,Tornado将默认返回一个200(OK)响应码。

 我们也可以自定义错误返回,只需在class中定义一个名为 write_error 的方法

比如

def write_error(self, status_code, **kwargs):
        self.write('%s' % status_code)

该段代码会将 错误code以字符串的形式返回回去

需要注意的是该方法是写在类中,也就是说该方法只能作用于一个视图。

本章我们来学习 Tornado 支持的请求方式

请求方式

Tornado支持任何合法的HTTP请求(GETPOSTPUTDELETEHEADOPTIONS)。你可以非常容易地定义上述任一种方法的行为,只需要在 RequestHandler 类中使用同名的方法。(也就是在视图中定义同名的方法)

关于请求方式对应的操作和返回码可以参考 http://www.runoob.com/w3cnote/restful-architecture.html 不过具体还是要与业务进行匹配。

状态码

使用RequestHandler类的set_status()方法显式地设置HTTP状态码。然而,你需要记住在某些情况下,Tornado会自动地设置HTTP状态码。

比如如果访问一个无法匹配的路由,则会返回404报错,而且状态码是404

但是我们在代码中没有指定返回的状态码,常用的有:

  404 Not Found

  Tornado会在HTTP请求的路径无法匹配任何RequestHandler类相对应的模式时返回404(Not Found)响应码。

  400 Bad Request

  如果你调用了一个没有默认值的get_argument函数,并且没有发现给定名称的参数,Tornado将自动返回一个400(Bad Request)响应码。

  405 Method Not Allowed

  如果传入的请求使用了RequestHandler中没有定义的HTTP方法(比如,一个POST请求,但是处理函数中只有定义了get方法),Tornado将返回一个405(Methos Not Allowed)响应码。

  500 Internal Server Error

  当程序遇到任何不能让其退出的错误时,Tornado将返回500(Internal Server Error)响应码。你代码中任何没有捕获的异常也会导致500响应码。

  200 OK

  如果响应成功,并且没有其他返回码被设置,Tornado将默认返回一个200(OK)响应码。

 我们也可以自定义错误返回,只需在class中定义一个名为 write_error 的方法

比如

def write_error(self, status_code, **kwargs):
        self.write('%s' % status_code)

该段代码会将 错误code以字符串的形式返回回去

需要注意的是该方法是写在类中,也就是说该方法只能作用于一个视图。

おすすめ

転載: www.cnblogs.com/hanzeng1993/p/11281008.html