假设我在python Tornado框架中有一个非常简单的Web应用程序,只有一个端点。我感兴趣的只是返回在启动服务器之前计算的值。稍微修改一下 https://www.tornadoweb.org/en/stable/index.html 的例子就可以了。
handler.py
import tornado.web
class MainHandler(tornado.web.RequestHandler):
def get(self):
self.write('I want to return var `expensive_value`')
main.py
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()
当运行python main.py
并向端点发送请求时,它当然只返回一个字符串。但我想返回expensive_value
的实际值。目前我知道这个问题有两种解决方案。
1. 在处理程序中使用全局变量
handler.py
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)
main.py
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()
拥有一个全局变量并从其他模块设置其值对我来说听起来像是一种反模式。
2. 在处理程序中使用初始化方法
handler.py
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)
main.py
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()
这个解决方案更好。但是initialize
每个请求都会调用该方法。我意识到这样做的开销相当小,但我认为这可能会误导代码的潜在读者,因为expensive_value
永远不会改变。
总结
这两种解决方案都有效。但我不喜欢它们中的任何一个,似乎我缺少一些龙卷风功能。解决这个问题的pythonic方法是什么?
例如,我相信 Flask 有app.config
字典可以在处理程序中访问,这似乎是一个很好的解决方案,因为它确实是应用程序的配置expensive_value
。但我不知道龙卷风中有什么类似的东西。
处理程序可以访问self.application.settings
,这是一个字典,其中包含传递给Application
构造函数的其他参数。
因此,您可以像这样将expensive_value
直接传递给Application
类:
def make_app(parameter):
return tornado.web.Application(
[
(r"/", MainHandler),
],
expensive_value=parameter
)
并在任何处理程序中访问此值,如下所示:
def initialize(self):
self.expensive_value = self.application.settings.get('expensive_value')