曾经,用惯了python print命令的人,惊叹于python语法的精简;后来,用过了tornado、django等web开发框架,不得不佩服当初开发这些框架的人们。于是,我们开始使用它们的框架==》一个被重复了无数次的配置。那么问题来了,如果我很多次的完成某个领域的类似问题,我需要每次都去重复的配置吗?偶然间,我在想,能否在它们的框架上继续精简,以在某种用途上实现更加高效的开发。web框架的使用,包含server端和browser端。本次改装采用tornado框架。
首先,我们从接触python第一天的print(“Helloworld!”)开始。
干货开始:
首先我们来看一段精简的tornado代码:
import tornado.ioloop import tornado.web class MainHandler(tornado.web.RequestHandler): def get(self): self.write("Hello, world") application = tornado.web.Application([ (r"/index", MainHandler), ]) if __name__ == "__main__": application.listen(8888) tornado.ioloop.IOLoop.instance().start()
上述代码实现了在浏览器端输出hello,world!从上到下各部分的功能依次是:模块导入、业务处理类、路由映射、服务端开启。
本博文详细介绍模块的改装,更多tornado基础内容,详见下面链接:
上述代码虽然实现了在浏览器的输出内容,但是有很多不足之处:
- 每次打印一个东西,用户都需要写那么多代码
- 只能支持字符串打印,不能支持字典、列表、元组等常见数据结构
我们知道,python的print用法很简单,只需要我们将需要打印的东西放在print内部,即可用一句话打印出我们想要的结果来。那么问题来了,有没有一种简单的方法能够类似print使用一句话在浏览器上打印出我们想要的任何数据结构?
答案当然是:有。没错,你没有看错,接下来,博主来带你一步一步实现这个神奇的功能。
模块导入:
import tornado.ioloop import tornado.web import json
和之前有所不同的是,我们增加了json模块,聪明你你应该已经猜到了,博主是想通过json实现对不同数据结构的支持,是的啊,没错!有人说,这不算什么,我也会,好,别走,继续往下看!更多精彩,详见下文!
解决思路:
事实上,解决这个问题,其实很简单,核心问题就是如何通过传参方式,将用户需要打印的数据传递给self.write。
要解决这个问题,我们需要指定,tornado框架哪些地方可以进行传参,我们再来看一下tornado代码:
import tornado.ioloop import tornado.web class MainHandler(tornado.web.RequestHandler): def get(self): self.write("Hello, world") application = tornado.web.Application([ (r"/index", MainHandler), ]) if __name__ == "__main__": application.listen(8888) tornado.ioloop.IOLoop.instance().start()
我们采用由内向外的分析方法:
- 首先是self.write("Hello, world"),我们希望实现的结果是self.write(arg),arg代表用户传入的参数。而这个arg又要从哪里传入呢?包着它的是get方法,我们知道http协议的get方法可以通过在浏览器的url上通过问号加键值对的方式传入数据,可是难道用户每次打印数据都去浏览器的url输入吗?这明显不合理。因此get方法传参是行不通的。
- 接下来往外是业务处理类MainHandler,它能传参吗?我们知道类是可以通过init方法进行传参的,每次对类进行实例化的时候进行传参即可,我们需要自己定义个init方法并继承父类的init方法吗?可是问题是,这个MainHandler到底是在哪里进行实例化的,我们又需要在哪里对其进行传参呢?这里就到了tornado的源码部分,可是源码岂能随便去改,因此init方法传参也不合理。
- 为了遵循对源码封闭,对扩展开放的原则,其实tornado源码中已经为我们预留了位置,这就是传说中的initialize方法,也就是人们所说的“钩子”。
initialize方法如何使用呢?
首先需要自己定义一个类,用于复写源码中的initialize方法,因为源码中initialize方法其实什么事也没干。
class BaseHandler(tornado.web.RequestHandler): def initialize(self,arg): self.arg=arg
在这里,我们为initialize方法传入了参数arg,并给类增加属性self.arg=arg,这样做的目的是为了继承它的子类能够使用这个参数,这就是面向对象的封装,参数传递一次,这个类的子类的所有方法都是可以使用的。没错,聪明的你又发现了,等会如果我们的MainHandler类只要继承了BaseHandler类,那么MainHandler类的get方法中,就可以使用参数self.arg。下面就是MainHandler类的改装。
class MainHandler(BaseHandler): def get(self): self.write(self.arg)
做到这里,你可能已经发现:好吧,initialize方法传参我会了,可是用户输入的参数又是如何传入initialize方法呢?
我们发现唯一跟MainHandler类有关的,就是路由映射了,它是一个元组,元组第0个元素是正则匹配,第一个元素是业务处理类的类名,于是博主就想,这里能否接收第三个参数呢?能否进行传参呢?来看一眼路由映射:
application = tornado.web.Application([ (r"/index", MainHandler, ), ])
经过博主对Application类源码进行分析,果然,它是支持传第三个参数的:
源码截图:
源码中spec即每一个元组。更多源码问题,欢迎读者在下方评论,与博主进行探讨。经过博主多方咨询探讨,最终得出结论,第三个参数支持字典格式进行传递(源码中是**kwargs)。
路由映射改装代码:
application = tornado.web.Application([ (r"/index", MainHandler, dict(arg=json.dumps(request))), ])
同时,为了支持常见数据结构,博主采用json进行序列化数据。
以上就是整个改装流程,下面附整体代码模块和该模块的使用方法。
import tornado.ioloop import tornado.web import json class BaseHandler(tornado.web.RequestHandler): def initialize(self,arg): self.arg=arg class MainHandler(BaseHandler): def get(self): self.write(self.arg) def print_to_browser(request,port=8888): application = tornado.web.Application([ (r"/", MainHandler, dict(arg=json.dumps(request))), ]) application.listen(port) tornado.ioloop.IOLoop.instance().start()
上述模块是博主原创,封装了tornado路由映射、服务端开启,支持传参,第一个参数即需要打印的内容,第二个参数为可选参数,默认端口8888,如果用户有传递,则以用户输入的参数为准。
鉴于博主有文章被多家网站盗用,鉴于此,原谅博主在此插播声明一条:本博文为博主原创,转账请注明原文链接:http://www.cnblogs.com/wanghzh/p/5869336.html,谢谢合作!
模块使用:
#!/usr/bin/env python # -*- coding: utf-8 -*- from my.mytornado import print_to_browser #导入上述模块 print_to_browser([1,2,3,4,])
结果示例截图:
上述使用方法是不是同python自带的print一样简单呢,支持字符串、字典、列表、元组等数据结构。默认域名:127.0.0.1,默认端口:8888。用户只需开启浏览器客户端即可(127.0.0.1:8888)。
如果您觉得本文对您有参考价值,,欢迎帮博主点下文章下方的推荐,非常谢谢!