Python Web 框架:Tornado

Tornado:Tornado web server

1.Tornado

  • Tornado:python編寫的web服務(wù)器兼web應(yīng)用框架

1.Tornado的優(yōu)勢(shì)

  • 輕量級(jí)web框架
  • 異步非阻塞IO處理方式
  • 出色的抗負(fù)載能力
  • 優(yōu)異的處理性能阀捅,不依賴多進(jìn)程/多線程故硅,一定程度上解決C10K問(wèn)題
  • WSGI全棧替代產(chǎn)品,推薦同時(shí)使用其web框架和HTTP服務(wù)器

2.Tornado VS Django

  • Django:重量級(jí)web框架糯崎,功能大而全拓哟,注重高效開發(fā)
  • 內(nèi)置管理后臺(tái)
  • 內(nèi)置封裝完善的ORM操作
  • session功能
  • 后臺(tái)管理
  • 缺陷:高耦合
  • Tornado:輕量級(jí)web框架想许,功能少而精,注重性能優(yōu)越
  • HTTP服務(wù)器
  • 異步編程
  • WebSocket
  • 缺陷:入門門檻較高

2.安裝

輸入命令:

pip install tornado

備注:
Tornado應(yīng)該運(yùn)行在類Unix平臺(tái)断序,為了達(dá)到最佳的性能和擴(kuò)展性流纹,僅推薦Linux和BSD(充分利用Linux的epoll工具和BSD的kqueue達(dá)到高性能處理的目的)

3.使用

1.Tornado入門程序 - (一)

#-*- coding:utf-8 -*-
import tornado.web
import tornado.ioloop


#定義處理類型
class IndexHandler(tornado.web.RequestHandler):
    #添加一個(gè)處理get請(qǐng)求方式的方法
    def get(self):
        #向響應(yīng)中,添加數(shù)據(jù)
        self.write('好看的皮囊千篇一律违诗,有趣的靈魂萬(wàn)里挑一漱凝。')

if __name__ == '__main__':
    #創(chuàng)建一個(gè)應(yīng)用對(duì)象
    app = tornado.web.Application([(r'/',IndexHandler)])
    #綁定一個(gè)監(jiān)聽端口
    app.listen(8888)
    #啟動(dòng)web程序,開始監(jiān)聽端口的連接
    tornado.ioloop.IOLoop.current().start()

1 .在pycharm中直接運(yùn)行代碼
2 .如果是在ubuntu诸迟,在命令窗口輸入

python 文件名.py
使用瀏覽器訪問(wèn)

4.Tornado 代碼解析

1.入門程序代碼解析

  • tornado.web:tornado的基礎(chǔ)web框架

    • RequestHandler封裝對(duì)請(qǐng)求處理的所有信息和處理方法
    • get/post/..:封裝對(duì)應(yīng)的請(qǐng)求方式
    • write():封裝響應(yīng)信息茸炒,寫響應(yīng)信息的一個(gè)方法
  • tornado.ioloop:核心io循環(huán)模塊,封裝linux的epoll和BSD的kqueue阵苇, tornado高性能處理的核心壁公。

    • current()返回當(dāng)前線程的IOLoop實(shí)例對(duì)象
    • start()啟動(dòng)IOLoop實(shí)力對(duì)象的IO循環(huán)器瘪,開啟監(jiān)聽

2.httpserver底層處理

  • httpserver監(jiān)聽端口
    • tornado.httpserver.HTTPServer(app)
    • httpserver.listen(port)
  • httpserver實(shí)現(xiàn)多進(jìn)程操作
    • tornado.httpserver.HTTPServer(app)
    • httpserver.bind(port)
    • httpserver.start(0/None/<0/num)
# -*- coding:utf-8 -*-
from tornado.web import Application,RequestHandler
from tornado.ioloop import IOLoop
from tornado.httpserver import HTTPServer

class IndexHandler(RequestHandler):
    def get(self):
        self.write('給自己一點(diǎn)時(shí)間拥褂,理清所有的荒唐與期望。')

if __name__ == '__main__':
    app = Application([(r'/',IndexHandler)])
    http_server = HTTPServer(app)
    #最原始的方式
    http_server.bind(8888)
    http_server.start(0)

    #啟動(dòng)Ioloop輪循監(jiān)聽
    IOLoop.current().start()
同時(shí)打開兩個(gè)窗口測(cè)試發(fā)現(xiàn)實(shí)現(xiàn)了多進(jìn)程

3.options配置

  • 全局配置
tornado.options.define(
    name, default, type, multiple, help
)
  • 命令行參數(shù)轉(zhuǎn)換
tornado.options.parse_command_line()
#-*- coding:utf-8 -*-

from tornado.web import RequestHandler,Application
from tornado.ioloop import IOLoop
from tornado.httpserver import HTTPServer
import tornado.options

#定義變量
tornado.options.define('port',default=8000,type=int,help="this is the port >for application")

class IndexHandler(RequestHandler):
   def get(self):
       self.write('我們既然改變不了規(guī)則男翰,那就做到最好')

if __name__ == '__main__':
   app = Application([(r'/',IndexHandler)])
   tornado.options.parse_command_line()

   http_server = HTTPServer(app)
   http_server.bind(tornado.options.options.port)
   http_server.start(1)
   #啟動(dòng)IOLoop輪循監(jiān)聽
   IOLoop.current().start()
通過(guò)命令窗口輸入port來(lái)訪問(wèn)

通過(guò)使用我們命令窗口設(shè)定的port進(jìn)行訪問(wèn)
  • 配置文件
#即在當(dāng)前py文件目錄創(chuàng)建config文件快耿,并在py代碼中加入以下代碼囊陡,
tornado.options.parse_config_file("./config")
  • 配置模塊:跟配置文件類似

4.application配置

  • 程序調(diào)試之debug配置
#自動(dòng)重啟+取消緩存模板+取消緩存靜態(tài)文件+提供追蹤信息
tornado.web.Application([(..)], debug=True)

注:開發(fā)之初可以設(shè)置debug=True方便調(diào)試,開發(fā)完畢改為False.
  • 路由信息初始化參數(shù)配置
tonado.web.Application([(r””, Handler, {k:v})])
def initialize(self, k)
#-*- coding:utf-8 -*-
from tornado.web import RequestHandler,Application
from tornado.ioloop import IOLoop
from tornado.httpserver import HTTPServer
from tornado.options import define,options

define('port',default=8000,type=int)

class IndexHandler(RequestHandler):
   def get(self):
       self.write('get-->先自沉穩(wěn)润努,而后愛人')


class ArticleHandler(RequestHandler):
   def initialize(self,title):
       print('-->initialize()')
       self.title = title

   def get(self):
       self.write('你正在查看文章:%s'% self.title)

if __name__ == '__main__':
   options.parse_command_line()

   app = Application([(r'/',IndexHandler),(r'/article',ArticleHandler,{'title':'你>希望自己成為什么樣的人关斜,最終就會(huì)成為那樣的人。'})],debug=True)
   http_server = HTTPServer(app)
   http_server.bind(options.port)
   http_server.start(1)

   IOLoop.current().start()
輸入路徑铺浇,顯示結(jié)果
  • 路由名稱設(shè)置及反解析
#名稱設(shè)置
tornado.web.Application([
    url(r””, handler, {k,v}, name=“”)
])

#反解析操作
reverse_url(name)
# -*- coding:utf-8 -*-

from tornado.web import Application, RequestHandler, url
from tornado.ioloop import IOLoop
from tornado.httpserver import HTTPServer
from tornado.options import options,define

define("port", default=8000, type=int)

class IndexHandler(RequestHandler):

   def get(self):
       self.write("<a href='"+self.reverse_url("login")+"'>用戶登錄</a>")


class RegistHandler(RequestHandler):
   def initialize(self, title):
       self.title = title

   def get(self):
       self.write("注冊(cè)業(yè)務(wù)處理:" + str(self.title))


class LoginHandler(RequestHandler):
   def get(self):
       self.write("用戶登錄頁(yè)面展示")

   def post(self):
      self.write("用戶登錄功能處理")


if __name__ == "__main__":
   app = Application(
       [
           (r"/", IndexHandler),
           (r"/regist", RegistHandler, {"title": "會(huì)員注冊(cè)"}),
           url(r"/login", LoginHandler, name="login"),
       ]
   )

   http_server = HTTPServer(app)
   http_server.listen(8000)

   IOLoop.current().start()
瀏覽器顯示結(jié)果

5.參數(shù)傳遞

  • get方式傳遞參數(shù)
get_query_arguments(name,default=_ARG_DEFAULT,strip=True)
get_query_argument(name ,strip=True)
# -*- coding:utf-8 -*-

from tornado.web import Application, RequestHandler
from tornado.ioloop import IOLoop
from tornado.httpserver import HTTPServer
from tornado.options import options, define

define("port", default=8000, type=int)


class IndexHandler(RequestHandler):

   def get(self):
       # 獲取get方式傳遞的參數(shù)
       username = self.get_query_argument("username")
       usernames = self.get_query_arguments("username")

       print (username)
       print (usernames)

if __name__ == "__main__":
   app = Application([(r"/", IndexHandler)])

   app.listen(8000)

   IOLoop.current().start()
  • post方式傳遞參數(shù)
get_body_arguments(name, default=_ARG_DEFAULT,strip=True)
get_body_argument(name ,strip=True)
  • 混合方式
get_arguments(..)/get_argument(..)
# -*- coding:utf-8 -*-

from tornado.web import Application, RequestHandler
from tornado.ioloop import IOLoop
from tornado.options import options, define

define("port", default=8000, type=int)


class IndexHandler(RequestHandler):

   def get(self):
       # 獲取get方式的參數(shù)
       user = self.get_argument("user")
       print("get方式獲取參數(shù):" + str(user))

   def post(self):
       # 獲取post方式的參數(shù)
       user = self.get_argument("user")
       print("post方式獲取參數(shù):" + user.encode("utf-8"))


if __name__ == "__main__":
   app = Application([(r"/", IndexHandler)])
   app.listen(8888)
   IOLoop.current().start()
使用谷歌瀏覽器的應(yīng)用postman可以看到get和post請(qǐng)求
  • 其他參數(shù)
通過(guò)request獲取參數(shù)數(shù)據(jù)
method/host/uri/path/query/version/headers/body/remote_ip/files

6.響應(yīng)頭設(shè)置

  • set_header(name, value)
  • set_default_headers(self)
  • add_header(name, value)
  • clear_header(name)
# -*- coding:utf-8 -*-

from tornado.web import Application, RequestHandler
from tornado.ioloop import IOLoop
from tornado.options import options, define

define("port", default=8000, type=int)


class IndexHandler(RequestHandler):
    def set_default_headers(self):
        # 第二種響應(yīng)頭設(shè)置方式
        print("---------> 響應(yīng)頭set_default_headers()執(zhí)行")
        self.set_header("Content-type", "application/json; charset=utf-8")
        self.set_header("js", "zj")

    def get(self):
        # 第一種操作響應(yīng)頭的方式:
        # self.set_header("Content-type", "application/json")
        print("---------->get方法執(zhí)行")
        self.write("{'簡(jiǎn)書':'知幾'}")
        self.set_header("jianshu", "zhiji")


if __name__ == "__main__":
    app = Application([(r"/", IndexHandler)])

    app.listen(8000)

    IOLoop.current().start()
瀏覽器顯示結(jié)果

7.cookie操作

  • cookies
  • set/get_cookie(name, value)

  • set/get_secure_cookie(name, value)

  • clear_cookie(name)

  • clear_all_cookie()

# -*- coding:utf-8 -*-

from tornado.web import Application, RequestHandler
from tornado.ioloop import IOLoop
from tornado.options import options, define

define("port", default=8000, type=int)


class IndexHandler(RequestHandler):

    def get(self):
        self.write("hello jianshu.com")

        self.set_cookie("loginuser", "admin")

        print self.get_cookie("loginuser")

        print self.cookies


if __name__ == "__main__":
    app = Application([(r"/", IndexHandler)])

    app.listen(8000)

    IOLoop.current().start()
可以看到cookie已生效

8.響應(yīng)錯(cuò)誤碼 & 錯(cuò)誤描述

  • set_status(status_code, reason=None)
# -*- coding:utf-8 -*-

from tornado.web import Application, RequestHandler
from tornado.ioloop import IOLoop
from tornado.options import options, define

define("port", default=8000, type=int)


class IndexHandler(RequestHandler):

    def get(self):
        self.write("hello 簡(jiǎn)書")
        # self.set_status(404) # 標(biāo)準(zhǔn)錯(cuò)誤碼 無(wú)描述使用默認(rèn)描述
        self.set_status(201, reason="zhiji is cool") # 自定義錯(cuò)誤碼痢畜,設(shè)置reason描述
        # self.set_status(230) # 自定義錯(cuò)誤碼,無(wú)reason 報(bào)錯(cuò)

if __name__ == "__main__":
    app = Application([(r"/", IndexHandler)])

    app.listen(8000)

    IOLoop.current().start()
通過(guò)審查元素我們可以測(cè)試定義的錯(cuò)誤碼有沒(méi)有生效
  • send_error(status_code, reason=None)
# -*- coding:utf-8 -*-

from tornado.web import Application, RequestHandler
from tornado.ioloop import IOLoop
from tornado.options import options, define

define("port", default=8000, type=int)


class IndexHandler(RequestHandler):

    def get(self):
        self.write("hello qikuedu.com")

        self.send_error(500, reason="出錯(cuò)啦出錯(cuò)啦")

if __name__ == "__main__":
    app = Application([(r"/", IndexHandler)])

    app.listen(8000)

    IOLoop.current().start()
瀏覽器顯示結(jié)果
  • write_error(self, status_code, **kw)
# -*- coding:utf-8 -*-

from tornado.web import Application, RequestHandler
from tornado.ioloop import IOLoop
from tornado.options import options, define

define("port", default=8000, type=int)


class IndexHandler(RequestHandler):

    def get(self):
        self.write("hello qikuedu.com")

        self.send_error(404, msg="頁(yè)面丟失", info="家里服務(wù)器搞對(duì)象去了")

    def write_error(self, status_code, **kwargs):
        self.write("<h1>出錯(cuò)啦,工程師MM正在趕來(lái)的途中...</h1>")
        self.write("<p>錯(cuò)誤信息:%s</p>" % kwargs["msg"])
        self.write("<p>錯(cuò)誤描述:%s</p>" % kwargs["info"])


if __name__ == "__main__":
    app = Application([(r"/", IndexHandler)])

    app.listen(8000)

    IOLoop.current().start()
瀏覽器顯示結(jié)果
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末丁稀,一起剝皮案震驚了整個(gè)濱河市吼拥,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌线衫,老刑警劉巖凿可,帶你破解...
    沈念sama閱讀 207,113評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異授账,居然都是意外死亡枯跑,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門白热,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)敛助,“玉大人,你說(shuō)我怎么就攤上這事屋确∧苫鳎” “怎么了?”我有些...
    開封第一講書人閱讀 153,340評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵攻臀,是天一觀的道長(zhǎng)焕数。 經(jīng)常有香客問(wèn)我,道長(zhǎng)刨啸,這世上最難降的妖魔是什么堡赔? 我笑而不...
    開封第一講書人閱讀 55,449評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮呜投,結(jié)果婚禮上加匈,老公的妹妹穿的比我還像新娘。我一直安慰自己仑荐,他們只是感情好雕拼,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評(píng)論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著粘招,像睡著了一般啥寇。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上洒扎,一...
    開封第一講書人閱讀 49,166評(píng)論 1 284
  • 那天辑甜,我揣著相機(jī)與錄音,去河邊找鬼袍冷。 笑死磷醋,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的胡诗。 我是一名探鬼主播邓线,決...
    沈念sama閱讀 38,442評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼淌友,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了骇陈?” 一聲冷哼從身側(cè)響起震庭,我...
    開封第一講書人閱讀 37,105評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎你雌,沒(méi)想到半個(gè)月后器联,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,601評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡婿崭,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評(píng)論 2 325
  • 正文 我和宋清朗相戀三年拨拓,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片氓栈。...
    茶點(diǎn)故事閱讀 38,161評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡千元,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出颤绕,到底是詐尸還是另有隱情,我是刑警寧澤祟身,帶...
    沈念sama閱讀 33,792評(píng)論 4 323
  • 正文 年R本政府宣布奥务,位于F島的核電站,受9級(jí)特大地震影響袜硫,放射性物質(zhì)發(fā)生泄漏氯葬。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評(píng)論 3 307
  • 文/蒙蒙 一婉陷、第九天 我趴在偏房一處隱蔽的房頂上張望帚称。 院中可真熱鬧,春花似錦秽澳、人聲如沸闯睹。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)楼吃。三九已至,卻和暖如春妄讯,著一層夾襖步出監(jiān)牢的瞬間孩锡,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工亥贸, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留躬窜,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,618評(píng)論 2 355
  • 正文 我出身青樓炕置,卻偏偏與公主長(zhǎng)得像荣挨,于是被迫代替她去往敵國(guó)和親男韧。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評(píng)論 2 344

推薦閱讀更多精彩內(nèi)容

  • 譯者說(shuō) Tornado 4.3于2015年11月6日發(fā)布垦沉,該版本正式支持Python3.5的async/await...
    TaoBeier閱讀 3,030評(píng)論 0 10
  • tornado框架是一款相較于其他web framework處理服務(wù)器性能問(wèn)題更加強(qiáng)健的輕量級(jí)的強(qiáng)大的Pytho...
    惡人未滿閱讀 7,109評(píng)論 3 10
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理煌抒,服務(wù)發(fā)現(xiàn),斷路器厕倍,智...
    卡卡羅2017閱讀 134,601評(píng)論 18 139
  • tornado簡(jiǎn)單的WEB服務(wù) 在本節(jié)開始前寡壮,確認(rèn)你有一定的python基礎(chǔ),并且pycharm和服務(wù)器環(huán)境搭建完...
    蝸牛加餐了閱讀 1,597評(píng)論 0 0
  • 今天是情人節(jié)讹弯,與我沒(méi)什么關(guān)系况既。我比較在意的是明天,2月15日组民,研究生考試成績(jī)發(fā)布的日子棒仍。 一直抱著一種什么樣的情緒...
    米鹵蛋愛甜點(diǎn)山閱讀 287評(píng)論 0 1