参数化 Tornado RequestHandler

Mic*_*ppi 4 python web-services tornado

假设我在 python Tornado 框架中有一个非常简单的 Web 应用程序,它只有一个端点。我感兴趣的只是返回一个在启动服务器之前计算的值。从https://www.tornadoweb.org/en/stable/index.html稍微修改的例子就可以了。

处理程序

import tornado.web


class MainHandler(tornado.web.RequestHandler):
    def get(self):
        self.write('I want to return var `expensive_value`')
Run Code Online (Sandbox Code Playgroud)

主文件

import tornado.ioloop
import tornado.web


def make_app():
    return tornado.web.Application([
        (r"/", MainHandler),
    ])

if __name__ == "__main__":
    # calculate some var here before starting the server
    expensive_value = 'value from long_calculation()'
    app = make_app()
    app.listen(8888)
    tornado.ioloop.IOLoop.current().start()
Run Code Online (Sandbox Code Playgroud)

当运行python main.py并向端点发送请求时,它当然只返回一个字符串。但我想返回expensive_value. 目前我知道该问题的两种解决方案。

1.在handler中使用全局变量

处理程序

import tornado.web


global_variable = None


def setter(val):
    global global_variable
    global_variable = val


class MainHandler(tornado.web.RequestHandler):
    def get(self):
        self.write(global_variable)

Run Code Online (Sandbox Code Playgroud)

主文件

import tornado.ioloop
import tornado.web

from handler import MainHandler, setter


def make_app():
    return tornado.web.Application([
        (r"/", MainHandler),
    ])


if __name__ == "__main__":
    expensive_value = 'value from long_calculation()'
    setter(expensive_value)
    app = make_app()
    app.listen(8888)
    tornado.ioloop.IOLoop.current().start()
Run Code Online (Sandbox Code Playgroud)

拥有一个全局变量并从其他一些模块设置它的值对我来说听起来像是一种反模式。

2.在处理程序中使用初始化方法

处理程序

import tornado.web


class MainHandler(tornado.web.RequestHandler):
    def initialize(self, expensive_value):
        self.expensive_value = expensive_value

    def get(self):
        self.write(self.expensive_value)
Run Code Online (Sandbox Code Playgroud)

主文件

import tornado.ioloop
import tornado.web

from handler import MainHandler


def make_app(parameter):
    return tornado.web.Application([
        (r"/", MainHandler, {'expensive_value': parameter}),
    ])


if __name__ == "__main__":
    expensive_value = 'value from long_calculation()'
    app = make_app(expensive_value)
    app.listen(8888)
    tornado.ioloop.IOLoop.current().start()
Run Code Online (Sandbox Code Playgroud)

这个解决方案更好。但是initialize每个请求都会调用方法。我意识到这样做的开销会相当小,但我认为它可能会误导潜在的代码读者,因为它expensive_value永远不会改变。

概括

这两种解决方案都有效。但我不喜欢它们中的任何一个,而且我似乎缺少一些 Tornado 功能。解决这个问题的pythonic方法是什么?

例如,我相信 Flask 有app.config可以在处理程序中访问的字典,这似乎是一个很好的解决方案,因为expensive_value它确实是应用程序的配置。但我不知道 Tornado 中有任何类似的东西。

xyr*_*res 5

处理程序可以访问self.application.settingswhich 是一个包含传递给Application构造函数的附加参数的字典。

所以你可以像这样expensive_value直接传递给Application类:

def make_app(parameter):
    return tornado.web.Application(
        [
            (r"/", MainHandler),
        ],

        expensive_value=parameter
    )
Run Code Online (Sandbox Code Playgroud)

并在任何处理程序中访问此值,如下所示:

def initialize(self):
    self.expensive_value = self.application.settings.get('expensive_value')
Run Code Online (Sandbox Code Playgroud)