2

假设我有以下简单的类:

import cherrypy
import os

class test:
  test_member = 0;
  def __init__(self):
    return
  def index(self):
    self.test_member = self.test_member + 1
    return str(self.test_member)
  index.exposed = True

conf = os.path.join(os.path.dirname(__file__), 'config.ini')

if __name__ == '__main__':
  # CherryPy always starts with app.root when trying to map request URIs
  # to objects, so we need to mount a request handler root. A request
  # to '/' will be mapped to HelloWorld().index().
  cherrypy.config.update({'server.socket_host': '0.0.0.0'})
  cherrypy.quickstart(test(), config=conf)
else:
  # This branch is for the test suite; you can ignore it.
  cherrypy.config.update({'server.socket_host': '0.0.0.0'})
  cherrypy.tree.mount(test(), config=conf)

因此,当我第一次打开索引页面时,我会返回 1,下一次返回 2,然后是 3、4,依此类推。我的问题是:

  • 这有什么大的危险,特别是线程和多人同时访问页面?
  • 为了防止出现问题,我是否必须在每次写入成员变量时以某种方式锁定它?
  • 如果我使用非基本数据类型作为成员(例如我自己的复杂类)而不是像整数这样简单的东西,有什么改变吗?

我不完全理解 CherryPy 的线程是如何工作的,我想我在这个简单的例子中担心的是,在一个线程上 test_member 可能等于一件事,而当从另一个线程访问时,它会完全不同。如果我遗漏了一些有据可查的东西,我提前道歉,但是一些谷歌搜索并没有真正找到我想要的东西。我知道对于这样一个简单的示例,有许多相对简单的路径可以解决这里的潜在问题(将变量的状态保留在数据库中,或者类似的东西),但这在我的实际用例中不起作用.

4

1 回答 1

4

那里有丢失更新的危险。只需设置值不需要锁定,因为替换实例变量相对于 GIL 是原子的(假设它不调用任何特殊方法等)。但是增加或使用更复杂的变量将需要不同的方案来使它们成为线程安全的。

CherryPy 中的共享访问通常与任何其他 Python 程序没有什么不同。与其在这里对所有这些选项进行长时间的重复,最好将您引导至http://effbot.org/zone/thread-synchronization.htm正如它所提到的,替换实例变量对于 GIL 而言可能是原子的,因此线程安全,但递增不是。

CherryPy 只在相反的方向添加了一些助手:当您不想共享时:cherrypy.requestand对象是为每个请求/响应新创建(并正确销毁)的——如果您想保留它,请cherrypy.response随意粘贴数据cherrypy.request.foo仅在请求期间。

于 2012-09-10T14:53:36.100 回答