global_counter = 0 connections = {} class SocketHandler(): currentid = 0 def open(self): global global_counter global connections currentid = global_counter global_counter += 1 connections[currentid] = self print "WebSocket " + str(currentid) + " opened" def on_close(self): global connections print "WebSocket " + str(currentid) + " closed" del connections[currentid]@H_502_3@我收到错误:
NameError: global name 'currentid' is not defined@H_502_3@在“打开”和“on_close”的行上,我打印我打开/关闭连接.我在课堂上定义了它,为什么不在范围内.另外,我已经读过使用全局变量很糟糕,但我没有看到解决方法.有人可以指出我应该做什么吗?谢谢.
解决方法
像行中的当前名称一样简单:
del connections[currentid]@H_502_3@在尝试全局模块范围之前,总是在本地函数范围中查找名称,然后在每个封闭的函数范围中查找(然后将内置函数视为最后的手段). currentid是一个类属性,在任何这些范围中都找不到.
要在Python中查找属性,您始终需要指定要查看的对象.虽然查找协议意味着对象不一定必须具有属性本身;属性查找将回退到您指定的对象的类(以及基类,如果涉及继承).
这样可行:
del connections[self.currentid]@H_502_3@但是,我不认为你的代码的其余部分正在做你认为的那样.这行在open方法中:
currentid = global_counter@H_502_3@不设置SocketHandler对象的currentid属性.分配给一个裸名称总是分配给一个局部变量,除非你明确地将它声明为全局(你似乎已经意识到这一点,因为你已经使用了global关键字).所以在open方法中,currentid是一个局部函数变量;它的值在open方法结束时丢失.
实际上,您的SocketHandler对象根本没有currentid属性(除非您没有向我们展示更多代码).将currentid = 0放在类块中并不会为所有SocketHandler实例提供currentid属性.它为SocketHandler类本身提供了一个属性currentntid;这就像def open(self):block在类对象上创建一个open属性(存储一个函数),而不是在每个单独的实例上.
在on_close方法中读取self.currentid将无法在对象self中找到currentid属性,因此Python将查看self类,即SocketHandler.该对象确实具有currentntid值,因此读取self.currentid的结果将为0,无论您以前是否在该SocketHandler上运行open.
如果您打算将currentid存储为每个SocketHandler中的实例变量,那么open中的行必须是:
self.currentid = global_counter@H_502_3@这将指定self引用的对象的currentid属性.然后,您还需要将方法中对currentid的所有其他引用更改为self.currentid.