上一篇中,提到了装饰器在tornado中的封装风格,今天就来实现一下。
首先需要了解tornado本来的风格,从一个最简单的helloworld开始。
import sys,tornado.ioloop,tornado.web,tornado
class Xroute(tornado.web.RequestHandler):
def get(self, path):
self.write("hello world,get")
#main
if __name__=="__main__":
port=8888
application = tornado.web.Application([(r"/(.*)", Xroute),])
if len(sys.argv)>1:
port = int(sys.argv[1])
application.listen(port)
tornado.ioloop.IOLoop.instance().start()
首先是定义一个requestHandler子类,里面可以实现get,post等方法,可以看到,回写页面也是靠这个对象的write方法。然后配置对应的URL(上例中默认是所有URL)传入到Application对象中去就OK了。这种风格的写法和javaEE中的servlet一样,实现一个servlet子类,重写doPost,doGet方法,然后在web.xml中配置对应路径,很显然过于麻烦了,特别是在多人合作的时候就显得比较复杂。体会过bottle的人应该对于那种即熟悉又陌生的装饰器风格印象深刻。下面我们就自己实现一个。
构思,上一篇中提到利用装饰器在装饰的时候会调用一次的特点可以收集所有业务函数和它的装饰器参数。难点在于如何构造函数的参数?获取页面参数,回写页面是需要RequestHandler参与的,所以我们必须把这个参数注入到业务函数中,剩下就是URL中截取的参数了,举个栗子:localhost:8888/book/medicine/49875,其中medicine是book的category,49875是书的id。这两个参数也是需要注入的。例如:
@Router.route(url=r"book/([a-z]+)/(\d+)",method=Router._GET|Router._POST)
def test3(req, categories, bookid):
#http://localhost:8888/book/medicine/49875
return "looking for a " + categories + " book." + "No." + bookid
好了确定了之后我们可以开始着手写装饰器类了(建立文件route.py)
#!/usr/bin/python
#coding=utf-8
import re
class Router(object):
'''dispather and decortor'''
_GET = 0x001
_POST = 0x002
_PUT = 0x004
_DELETE = 0x008
#mapper中存放的就是所有的业务方法,key为装饰器上面的url参数
mapper={}
#装饰器方法
@classmethod
def route(cls, **deco):
print Router.mapper
def foo(func):
#装饰器url如果不存在于mapper中,就将以下结构存入mapper中,例如
# “hello/(\w+)” : {
# "call" = func,
# "method" = "GET"
# }
url = deco.get('url') or '/'
if url not in Router.mapper:
method = deco.get('method') or Router._GET
mapper_node = {}
mapper_node['method'] = method
mapper_node['call'] = func
Router.mapper[url] = mapper_node
return func
return foo
#get,post方法需要注入requestHandler实例,所以在此需要得到它
@classmethod
def get(cls, path, reqhandler):
Router.emit(path, reqhandler, Router._GET)
@classmethod
def post(cls,path,reqhandler):
Router.emit(path, reqhandler, Router._POST)
#put,delete,head...可以套这种写法套下去
@classmethod
def emit(cls, path, reqhandler, method_flag):
mapper = Router.mapper
for urlExp in mapper:
m = re.match('^'+urlExp+'$',path)
#如果用户访问地址能够匹配mapper中url映射规则
if m:
#构造注入参数,首先是requestHandler,然后是路径参数
params = (reqhandler,)
for items in m.groups():
params+=(items,)
mapper_node = mapper.get(urlExp)
method = mapper_node.get('method')
#如果不匹配装饰器参数中method的规则就报405异常
if method_flag is not method_flag & method:
raise tornado.web.HTTPError(405)
try:
call = mapper_node.get('call')
#执行方法回写页面
reqhandler.write(call(*params))
except Exception,e:
print e
raise tornado.web.HTTPError(500)
break
else:
raise tornado.web.HTTPError(404)
#为什么在这里引入业务函数,因为如果装饰器的特性是在装饰的时候调用一次,
#如果一开始就引进来,装饰器一调用发现装饰器类都没有定义就会报错,所以
#需要把装饰器类定义出来了之后引人
from biz import *
结尾引入了上一篇中提到的封装的业务函数文件进来。也就是(biz.py):
#!/bin/python
#coding=utf-8
import sys,traceback,tornado.web,time
from route import Router
@Router.route(url=r"hello/([a-z]+)",method=Router._GET|Router._POST)
def test(req, who):
#http://localhost:8888/hello/billy
return "Hi," + who + "\n"
@Router.route(url=r"greetings/([a-z]+)",method=Router._GET)
def test2(req, who):
#http://localhost:8888/greetings/rowland
raise Exception("error")
@Router.route(url=r"book/([a-z]+)/(\d+)",method=Router._GET|Router._POST)
def test3(req, categories,bookid):
#http://localhost:8888/book/medicine/49875
return "look for" + categories + " book" + " No." + bookid + "\n"
最后改造一下入口文件serv.py
#!/usr/bin/python
#coding=utf-8
import sys,tornado.ioloop,tornado.web,tornado
from route import Router
class Xroute(tornado.web.RequestHandler):
def get(self, path):
Router.get(path, self)
def post(self, path):
Router.post(path, self)
#put,head,delete等等可以套这种写法套下去
#main
if __name__=="__main__":
port=8888
application = tornado.web.Application([(r"^/([^\.|]*)(?!\.\w+)$", Xroute),])
if len(sys.argv)>1:
port = int(sys.argv[1])
application.listen(port)
tornado.ioloop.IOLoop.instance().start()
接下来,我们可以实验一下
看来已经达到效果了,但是目前会有一个问题,就是假设我们的业务是比较耗时的,tornado的特性又是单线程的。就会发生排队!我们可以实验一下,首先将biz中的test方法加上sleep模拟业务耗时的操作
@Router.route(url=r"hello/([a-z]+)",method=Router._GET|Router._POST)
def test(req, who):
time.sleep(5)
return "Hi," + who + "\n"
然后同时访问5次,看看所需耗时是多少
#!/bin/sh
function bb_curl()
{
for i in {1..5} ;do
curl localhost:8888/hello/billy &
done
}
function bb_test()
{
bb_curl
wait
}
#main
time bb_test
足足25秒钟!
tornado的原理和google的V8,nodejs很像,是单进程单线程,主进程会睡在epoll上,注册EPOLLIN事件,由管道对象消费,收集回调函数到队列中,然后尝试写入epoll,EPOLLOUT,从而唤醒主线程,主线程负责执行回调队列中所有函数。这也是为什么它们都比较费CPU但却都并发能力很强,效率很高的原因。所以想要使用他们最好的方式是写异步api,所有的工作全部异步化。
首先这对于一般程序员来说要改变编码习惯,全异步的API很容易造成逻辑的混乱,调试的难度陡然加大,而且tornado的异步api没有nodejs完善。
如果我们想利用tornado超强的并发处理,又想使用传统编程呢?其实也是有办法的,下一章,咱们就使用多线程改造它!
- 大小: 6.2 KB
- 大小: 14.3 KB
分享到:
相关推荐
Python+java+websocket+SpringMVC实时监控数据库中的表-源代码 参考地址,看看是否需要: https://truedei.blog.csdn.net/article/details/105630679 https://blog.csdn.net/qq_17623363/article/details/100551961
Hessian 支持 Java、C++、Python 等多种语言,使得跨语言的服务调用成为可能。Hessian 的优点在于它能够自动序列化和反序列化对象,减少了网络传输的数据量,提高了服务调用的效率。在分布式系统中,Hessian 可以...
黑马程序员主要面向互联网技术领域,提供高质量的技术培训课程,包括但不限于Java、Python、大数据、人工智能等多个方向。所提供的《SpringMVC+MyBatis由浅入深全套视频教程》旨在帮助学习者全面掌握这两种框架的...
使用Eclipse构建Maven的SpringMVC项目 Eclipse 是一个功能强大且通用...使用 Eclipse 构建 Maven 的 SpringMVC 项目可以提高项目的开发效率和质量,解决了多人开发时 jar 版本不同的问题,提供了一个清晰的文件结构。
2. **Eclipse**:Eclipse不仅仅是一个IDE,它还拥有一个庞大的插件生态系统,支持多种语言的开发,如Java、Python、C++等。Eclipse提供了代码编辑、调试、版本控制等多种功能,极大地提高了开发效率。 3. **Maven**...
5. **Git版本控制**: 版本管理工具,`.gitignore`用于排除不需要版本化的文件。 6. **项目许可证**: 如MIT、Apache等,定义项目代码的使用权限。 7. **Python打包和安装**: `setup.py`用于创建Python包并管理依赖。 ...
MongoDB、MyBatis、MySQL、Netty、Nginx、Oracle、Python、RabbitMQ、 React、Redis、Spring、SpringBoot、SpringCloud、SpringMVC、Vue、Zookeeper、 并发编程、大数据、前端等。 程序员找工作面试题大集锦,收集...
【标题】:“基于Python绘制圣诞树源码.zip”是一个压缩包,其中包含了使用Python编程语言编写的绘制圣诞树的源代码。这个资源主要是为了教育目的,适合学生学习和下载使用,帮助他们理解Python编程的基本概念,以及...
5. **SVN**:版本控制系统,协同开发必备工具。 6. **SpringMVC**:Spring的Web MVC框架,处理Web请求。 7. **JPA**:Java Persistence API,Java标准的ORM规范。 8. **SpringData**:Spring提供的数据访问框架,...
Python爬虫数据可视化分析是一项将网络数据通过爬虫技术获取并进行处理,再利用可视化工具进行展示和解读的重要技能。本项目"Python爬虫数据可视化分析大作业"旨在通过实践来提升这一领域的综合能力。该项目涉及到的...
Python简易HTTP服务器主要依赖于Python内置的`http.server`模块,这是一个非常基础的Web服务器,可以用于快速地本地测试网页内容。在这个项目中,开发者扩展了这个基础功能,增加了WebSocket支持,使得它能够处理...
MongoDB、MyBatis、MySQL、Netty、Nginx、Oracle、Python、RabbitMQ、 React、Redis、Spring、SpringBoot、SpringCloud、SpringMVC、Vue、Zookeeper、 并发编程、大数据、前端等。search、HTML、HTTP、JavaScript、...
MongoDB、MyBatis、MySQL、Netty、Nginx、Oracle、Python、RabbitMQ、 React、Redis、Spring、SpringBoot、SpringCloud、SpringMVC、Vue、Zookeeper、 并发编程、大数据、前端等。 程序员找工作面试题大集锦,收集...
MongoDB、MyBatis、MySQL、Netty、Nginx、Oracle、Python、RabbitMQ、 React、Redis、Spring、SpringBoot、SpringCloud、SpringMVC、Vue、Zookeeper、 并发编程、大数据、前端等。 程序员找工作面试题大集锦,收集...
标题 "基于ssm开发的电力大数据,hadoop+python数据抓取.zip" 暗示了这个项目是关于使用SSM(Spring、SpringMVC、MyBatis)框架开发的一个电力大数据应用,其中包含了Hadoop和Python在数据抓取方面的实践。...
包括STM32、ESP8266、PHP、QT、Linux、iOS、C++、Java、python、web、C#、EDA、proteus、RTOS等项目的源码。 【项目质量】: 所有源码都经过严格测试,可以直接运行。 功能在确认正常工作后才上传。 【适用人群】...
本文主要探讨的是基于Python的Django框架在高校社团学生会管理系统的应用。首先,我们需要理解这个系统是为了解决高校社团和学生会日常管理工作中的问题,提高管理效率而设计的。它涉及到的技术栈包括Python、Django...
在项目实践中,学生不仅可以提升Python编程技能,还能深入理解TCP/IP网络协议,学习数据库设计和管理,以及如何使用版本控制工具(如Git)进行协作开发。通过这样的课程设计,学生将具备解决实际网络管理问题的能力...
MongoDB、MyBatis、MySQL、Netty、Nginx、Oracle、Python、RabbitMQ、 React、Redis、Spring、SpringBoot、SpringCloud、SpringMVC、Vue、Zookeeper、 并发编程、大数据、前端等。 程序员找工作面试题大集锦,收集...
MongoDB、MyBatis、MySQL、Netty、Nginx、Oracle、Python、RabbitMQ、 React、Redis、Spring、SpringBoot、SpringCloud、SpringMVC、Vue、Zookeeper、 并发编程、大数据、前端等。 程序员找工作面试题大集锦,收集...