python tornado_Python Tornado篇

Tornado既是一个web server,也是web framework。而它作为web server 采用的是asynchronous IO的网络模型,这是一种很高效的模型。

Tornado 和现在的主流 Web 服务器框架(包括大多数 Python 的框架)有着明显的区别:它是非阻塞式服务器,而且速度相当快。得利于其 非阻塞的方式和对 epoll 的运用,Tornado 每秒可以处理数以千计的连接,这意味着对于实时 Web 服务来说,Tornado 是一个理想的 Web 框架。

同步IO操作导致请求进程阻塞,知道IO操作完成;异步IO操作不导致请求进程阻塞。

在Python中,同步IO可以被李杰为一个被调用的IO函数会阻塞调用函数的执行,而异步IO则不会阻塞调用函数执行。

安装pip3 install tornado

tornado网站架构简单示例:

importtornado.ioloopimporttornado.webclassMainHandler(tornado.web.RequestHandler):defget(self):importtime

time.sleep(10)

self.write("Hello, world")classIndexHandler(tornado.web.RequestHandler):defget(self):

self.write("Index")

application=tornado.web.Application([

(r"/main", MainHandler),

(r"/index", IndexHandler),

])if __name__ == "__main__":

application.listen(8888)

tornado.ioloop.IOLoop.instance().start()

View Code

执行过程:

第一步:执行脚本,监听 8888 端口

第二步:浏览器客户端访问 /index  -->  http://127.0.0.1:8888/index

第三步:服务器接受请求,并交由对应的类处理该请求

第四步:类接受到请求之后,根据请求方式(post / get / delete ...)的不同调用并执行相应的方法

第五步:方法返回值的字符串内容发送浏览器

#__author: Administrator#date: 2017/3/10

importtornado.webclassMainHandler(tornado.web.RequestHandler):defget(self):

self.write("Hello, world")classLoginHandler(tornado.web.RequestHandler):defget(self):#5.获取用户请求相关信息

#self.get_cookie()

#v = self.get_argument('p')

#print(v)

#self.request 封装了用户发来的所有请求

#print(type(self.request))

#from tornado.httputil import HTTPServerRequest

#6. 额外相应内容

#self.set_cookie('k1','v1')

#self.set_header('h1','v1')

#4. 返回页面+模版引擎

#self.render('login.html')

#self.render('login.html',k1='v1')

#self.render('login.html',k1='v1',k2='v2')

self.render('login.html',**{'k1':'v1','k2':[1,2,3,4],'k3':{'name':'root','age':18}})#7. 重定向

#self.redirect('/index/')

def post(self, *args, **kwargs):

v= self.get_argument('user')print(v)

self.redirect('http://www.autohome.com.cn')importtor.uimethods as mtfrom tor importuimodules as md#8. 配置

settings ={'static_path': 'static','static_url_prefix': '/sss/','template_path':'templates','ui_methods': mt,'ui_modules': md,

}#1.生成路由规则

application = tornado.web.Application([(r"/index", MainHandler),(r"/login", LoginHandler),],**settings)if __name__ == "__main__":#2. 创建socket对象8888

#将socket对象添加到select或epoll中

application.listen(8888)#3. 将select或epoll开始死循环 While True:

tornado.ioloop.IOLoop.instance().start()

异步非阻塞两种做法:

importtornado.ioloopimporttornado.webfrom tornado importgenclassMainHandler(tornado.web.RequestHandler):

@gen.coroutinedefget(self):from tornado importhttpclient

http=httpclient.AsyncHTTPClient()yield http.fetch("http://www.google.com", self.done)def done(self, *args, **kwargs):

self.write('Main')

self.finish()classIndexHandler(tornado.web.RequestHandler):defget(self):

self.write("Index")

application=tornado.web.Application([

(r"/main", MainHandler),

(r"/index", IndexHandler),

])if __name__ == "__main__":

application.listen(8888)

tornado.ioloop.IOLoop.instance().start()

使用AsyncHTTPClient

httpclient.AsyncHTTPClient()是异步访问,fetch函数会在调用后立即返回而不用等待实际访问的完成,从而导致get()也会立刻执行完成。

当访问实际完成后,AsyncHTTPClient会调用callback参数指定的函数,这里可以任意写代码。

装饰器 + Future 从而实现Tornado的异步非阻塞

importtornado.ioloopimporttornado.webfrom tornado importgenfrom tornado.concurrent importFuture

future=NoneclassMainHandler(tornado.web.RequestHandler):

@gen.coroutinedefget(self):globalfuture

future=Future()

future.add_done_callback(self.done)yieldfuturedef done(self, *args, **kwargs):

self.write('Main')

self.finish()classIndexHandler(tornado.web.RequestHandler):defget(self):globalfuture

future.set_result(None)

self.write("Index")

application=tornado.web.Application([

(r"/main", MainHandler),

(r"/index", IndexHandler),

])if __name__ == "__main__":

application.listen(8888)

tornado.ioloop.IOLoop.instance().start()

使用Future

当发送GET请求时,由于方法被@gen.coroutine装饰且yield 一个 Future对象,那么Tornado会等待,等待用户向future对象中放置数据或者发送信号,如果获取到数据或信号之后,就开始执行done方法。

异步非阻塞体现在当在Tornaod等待用户向future对象中放置数据时,还可以处理其他请求。

注意:在等待用户向future对象中放置数据或信号时,此连接是不断开的。

自定义异步非阻塞框架

importsocketimportselectclassHttpRequest(object):"""用户封装用户请求信息"""

def __init__(self, content):""":param content:用户发送的请求数据:请求头和请求体"""self.content=content

self.header_bytes=bytes()

self.body_bytes=bytes()

self.header_dict={}

self.method= ""self.url= ""self.protocol= ""self.initialize()

self.initialize_headers()definitialize(self):

temp= self.content.split(b'\r\n\r\n', 1)if len(temp) == 1:

self.header_bytes+=tempelse:

h, b=temp

self.header_bytes+=h

self.body_bytes+=b

@propertydefheader_str(self):return str(self.header_bytes, encoding='utf-8')definitialize_headers(self):

headers= self.header_str.split('\r\n')

first_line= headers[0].split(' ')if len(first_line) == 3:

self.method, self.url, self.protocol= headers[0].split(' ')for line inheaders:

kv= line.split(':')if len(kv) == 2:

k, v=kv

self.header_dict[k]=vclassFuture(object):def __init__(self):

self.result=None

F=Nonedefmain(request):globalF

F=Future()returnFdefstop(request):globalF

F.result= b"xxxxxxxxxxxxx"

return "stop"

defindex(request):return "indexasdfasdfasdf"routers=[

('/main/',main),

('/index/',index),

('/stop/',stop),

]defrun():

sock=socket.socket(socket.AF_INET, socket.SOCK_STREAM)

sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR,1)

sock.bind(("127.0.0.1", 9999,))

sock.setblocking(False)

sock.listen(128)

inputs=[]

inputs.append(sock)

async_request_dict={#'socket': futrue

}whileTrue:

rlist,wlist,elist= select.select(inputs,[],[],0.05)for r inrlist:if r ==sock:"""新请求到来"""conn,addr=sock.accept()

conn.setblocking(False)

inputs.append(conn)else:"""客户端发来数据"""data= b""

whileTrue:try:

chunk= r.recv(1024)

data= data +chunkexceptException as e:

chunk=Noneif notchunk:break

#data进行处理:请求头和请求体

request =HttpRequest(data)print(request.url)print(request.method)print(request.body_bytes)#1. 请求头中获取url

#2. 去路由中匹配,获取指定的函数

#3. 执行函数,获取返回值

#4. 将返回值 r.sendall(b'alskdjalksdjf;asfd')

importre

flag=False

func=Nonefor route inrouters:ifre.match(route[0],request.url):

flag=True

func= route[1]break

ifflag:

result=func(request)ifisinstance(result,Future):

async_request_dict[r]=resultelse:

r.sendall(bytes(result,encoding='utf-8'))

inputs.remove(r)

r.close()else:

r.sendall(b"404")

inputs.remove(r)

r.close()for conn inasync_request_dict.keys():

future=async_request_dict[conn]iffuture.result:

conn.sendall(future.result)

conn.close()delasync_request_dict[conn]

inputs.remove(conn)if __name__ == '__main__':

run()

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值