- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
看这个单例实现:
if not hasattr(Singleton, "_instance"):
with Singleton._instance_lock:
if not hasattr(Singleton, "_instance"):
Singleton._instance = Singleton()
return Singleton._instance
似乎“Singleton._instance = ..”(类似于setattr)和hasattr 是原子的。或者 hasattr 不会因为 setattr 而导致崩溃。
但我找不到任何支持以上“似乎”的东西。
最佳答案
通常,前提是您对其调用操作的对象未实现 __getattr__
, __delattr__
或 __setattr__
hooks in python,那么是的,hasattr
, getattr
, delattr
and setattr
都是原子操作。
就 Python 线程而言,任何单独的字节码都是原子操作。 Python 评估循环在解释操作码时获取全局解释器锁 (GIL)。
您需要查看字节码以了解边界所在:
>>> def foo():
... if not hasattr(Singleton, "_instance"):
... with Singleton._instance_lock:
... if not hasattr(Singleton, "_instance"):
... Singleton._instance = Singleton()
... return Singleton._instance
...
>>> dis.dis(foo)
2 0 LOAD_GLOBAL 0 (hasattr)
3 LOAD_GLOBAL 1 (Singleton)
6 LOAD_CONST 1 ('_instance')
9 CALL_FUNCTION 2
12 POP_JUMP_IF_TRUE 64
3 15 LOAD_GLOBAL 1 (Singleton)
18 LOAD_ATTR 2 (_instance_lock)
21 SETUP_WITH 35 (to 59)
24 POP_TOP
4 25 LOAD_GLOBAL 0 (hasattr)
28 LOAD_GLOBAL 1 (Singleton)
31 LOAD_CONST 1 ('_instance')
34 CALL_FUNCTION 2
37 POP_JUMP_IF_TRUE 55
5 40 LOAD_GLOBAL 1 (Singleton)
43 CALL_FUNCTION 0
46 LOAD_GLOBAL 1 (Singleton)
49 STORE_ATTR 3 (_instance)
52 JUMP_FORWARD 0 (to 55)
>> 55 POP_BLOCK
56 LOAD_CONST 0 (None)
>> 59 WITH_CLEANUP
60 END_FINALLY
61 JUMP_FORWARD 0 (to 64)
6 >> 64 LOAD_GLOBAL 1 (Singleton)
67 LOAD_ATTR 3 (_instance)
70 RETURN_VALUE
故事并没有就此结束; hasattr
使用 getattr()
(测试异常),后者又可以调用 Python __getattr__
Hook 。同样,STORE_ATTR
操作码最终可能会调用 python __setattr__
Hook 实现。在这两种情况下,GIL 都会被再次释放。
对于默认实现(Singleton
不实现这些钩子(Hook)),操作是原子的,因为 Python C 代码处理整个操作而不会回退到 Python,因此评估循环(GIL 可能被释放)并再次锁定另一个线程)。
当然,您仍然可以处理在 object protocol operations 期间释放锁的自定义 C 库.那将是一件不寻常的事情。
关于python - getattr/setattr/hasattr/delattr 线程安全吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16872210/
如果我使用 setattr(p,'wrong_attr','value') 而不是 p=MyModel.objects.filter(WRONG_ATTRIBUTE=value) 我没有收到与不存在的
我有一个定义了多个方法的类。 import mat class Klass(object): @mat.sell(mat.CanSet): def method1(self):
我不知所措。 找不到任何其他有助于解决此问题的内容。 dta = {'type': "", 'content': ""} print dta >>>{'content': '', 'type': ''
这就是我的意思: >>> class Foo: pass >>> foo = Foo() >>> setattr(foo, "@%#$%", 10) >>> foo.@%#$% Syn
这个问题已经有答案了: What do lambda function closures capture? (8 个回答) 已关闭 3 年前。 我正在尝试创建一个 class适应django反对导出。
我想了解对象删除在 python 上的工作原理。这是一组非常简单的代码。 class A(object): def __init__(self): setattr(self,
我有一个类(bot),它有一个属性“health”;由于这个类有很多参数,并且我希望用户输入很多参数,所以我选择循环遍历 {param:explanation} 的字典,并为每个参数输入一个要设置的值
我应该将什么作为第一个参数“object”传递给函数 setattr(object, name, value) , 在当前模块上设置变量? 例如: setattr(object, "SOME_CONS
我试图制作一个可以一次性将多个属性分配给不同值的类。因此,我尝试修改 setattr 来执行此操作。 class hello(): def __setattr__(self,attr,valu
我想使用 setattr 更改列的因子水平. However, when the column is selected the standard data.table方式( dt[ , col] ),
为什么在附加或连接对象属性的列表或字典时未调用 __setattr__ 方法。方法描述“在尝试属性分配时调用。”会让我相信连接两个列表是可行的,但是 __setattr__ 方法仍然没有被调用。 cl
我想以编程方式更改类的某些属性。 我发现有setattr函数,但它没有按预期工作。 > obj.setattr('bar', 99) 'MyClass' object has no attribute
我正在创建一个 sigsum() 函数,它使用输入方程和输入变量求和。这是我到目前为止所拥有的: def sigsum(eqn, index, lower=0, upper=None, step=1)
我在 cython 中有一个 cdef 类,我想用 setattr 内置函数初始化它的字段。但是,当我这样做时出现执行错误: /path/.../cimul.cpython-34m.so in cim
我有一个模型类,它有一个引用 django DB 对象的属性。我想使用带有 setattr() 的单个 View 来更改此属性,我用它来更改此对象的任何属性。 问题是,我似乎无法通过堆栈传递对象实例。
我想使用 setattr 创建绘图: import numpy as np import matplotlib.pyplot as plt x = np.random.rand
我试图动态地向类添加方法,但我一直遇到错误,其中 self 未传递给新函数。例如: class Dummy(): def say_hi(self): print("hi") d
我正在编写的一个类需要使用存储 numpy 数组的变量名属性。我想为这些数组的切片赋值。我一直在使用 setattr,这样我就可以让属性名称有所不同。我为切片赋值的尝试是: class Dummy(o
我对 python 还很陌生,但是我需要与一些作为一堆 python 模块编写的软件进行交互(.py 文件,以防我错误地将它们识别为“模块”。)这个程序有一些非常有用的功能以及我确实无法破解的复杂功能
class Product(models.Model): name = models.CharField(max_length=50) desc = models.CharFiel
我是一名优秀的程序员,十分优秀!