- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我看到这被标记为“类变量和实例变量之间有什么区别?”的副本但是我不相信我在我的示例中使用了任何实例变量,我的类都没有 __init__
我正在以两种不同的方式编辑类变量并试图理解它们之间的区别, 不是类和实例变量之间的区别。
我试图了解仅使用 .var
和使用 .__class__.var
调用类变量之间的区别。我认为这与子类化有关,所以我编写了以下代码。
class Foo:
foo = 0
class Bar(Foo):
bar = 1
def print_class_vals(f, b):
""" prints both instant.var and instant.__class__.var for foo and bar"""
print("f.foo: {}, {} "
"b.foo: {}, {} "
"b.bar: {}, {} "
"".format(f.foo, f.__class__.foo,
b.foo, b.__class__.foo,
b.bar, b.__class__.bar))
f = Foo()
b = Bar()
print_class_vals(f, b)
Foo.foo += 1
print_class_vals(f, b)
Bar.foo += 1
print_class_vals(f, b)
Bar.bar += 1
print_class_vals(f, b)
输出如下:
f.foo: 0, 0, b.foo: 0, 0, b.bar: 1, 1
f.foo: 1, 1, b.foo: 1, 1, b.bar: 1, 1
f.foo: 1, 1, b.foo: 2, 2, b.bar: 1, 1
f.foo: 1, 1, b.foo: 2, 2, b.bar: 2, 2
我似乎找不到调用 inst.var
和 inst.__class__.var
之间的任何区别。它们有何不同,我应该在什么时候使用一个而不是另一个?
最佳答案
同时 Gabriel Reis's answer完美地解释了这种特殊情况,f.foo
和 f.__class__.foo
实际上 有区别 即使 foo
未被实例属性隐藏。
比较:
>>> class Foo:
... foo = 1
... def bar(self): pass
... baz = lambda self: None
>>> f = Foo()
>>> f.foo
1
>>> f.__class__.foo
1
>>> f.bar
<bound method Foo.bar of <__main__.Foo object at 0x11948cb00>>
>>> f.__class__.bar
<function __main__.Foo.bar(self)>
>>> f.bar()
>>> f.__class__.bar()
TypeError: bar() missing 1 required positional argument: 'self'
f.baz
也是如此。
不同之处在于,通过直接访问 f.__class__.foo
,您将绕过 descriptor protocol ,这是使方法、@property
和类似的东西起作用的东西。
如果你想要完整的细节,请阅读链接的 HOWTO,但简短的版本是它比 Gabriel 的回答说的要多一些:
Python will look up for a name (attribute) first in the instance namespace/dict. If it doesn't find there, then it will look up in the class namespace. If it still doesn't find there, then it will walk through the base classes respecting the MRO (method resolution order).
但是如果它在类命名空间(或任何基类)中找到它,并且它找到的是一个描述符(一个具有 __get__
方法的值),它会执行一个额外的步骤。细节取决于它是数据描述符还是非数据描述符(基本上,它是否也有一个__set__
方法),但简短的版本是,而不是给你值,它会对该值调用 __get__
并为您提供该值返回的内容。函数有一个返回绑定(bind)方法的__get__
方法;属性有一个调用属性get方法的__get__
方法;等
关于python - 类变量,instance.var 和 instance.__class__.var 的区别,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50311057/
在 Python 的文档中 __class__被描述为一个属性。在对象中 type (元类),__class__似乎是一种方法。 如果我们这样做: >>> class Foo: pas
我有一系列的类,它们都继承自共享一个基类的几个可能的父类中的一个。每个类都有一个基于基类 parameters 的类作用域 Dict[str, object] parameters from copy
这个问题在这里已经有了答案: Why doesn't 2.__add__(3) work in Python? (2 个答案) 关闭 3 年前。 整数是唯一不支持.__class__操作的类型吗?例
我最近正在单步调试 CPython 源代码,特别是查看 symbol table entry编译期间的一个类。 我遇到了 typedef struct _symtable_entry 的以下条目结构:
我想测试一个对象是否是一个类的实例,并且只有这个类(没有子类)。我可以这样做: obj.__class__ == Foo obj.__class__ is Foo type(obj) == Foo t
有以下服务器代码(我正在定义一个继承自套接字的类,并尝试使用该类调用客户端套接字上的发送方法): import socket class LogSocket(socket.socket): d
python 依赖于 cell 中的 __class__ 变量来进行 super() 调用。它从第一个堆栈帧中的free 变量获取此单元格。 奇怪的是,这个变量不在 locals() 中,而是在您从
我对设置对象的属性和在对象的 __class__ 属性上设置属性之间的区别感到有点困惑。粗略地说,obj.attr 与 obj.__class__.attr。它们有什么区别? 前者是在一个实例上设置一
在 pinax Userdict.py 中: def __getitem__(self, key): if key in self.data: return s
这个问题在这里已经有了答案: How to avoid infinite recursion with super()? (1 个回答) 关闭 5 年前。 我在 WTForms 中处理继承。我有下一
Django 源代码: def new_method_proxy(func): def inner(self, *args): if self._wrapped is empt
使用 __class__ 在类中创建新实例是个好主意吗? 以下代码是执行此操作的示例: from collections import namedtuple _Position = namedtupl
我希望能够做到: >>> class a(str): ... pass ... >>> b = a() >>> b.__class__ = str Traceback (most recent
我有一个关于 python 中的 __class__ 的问题。 文档说 __class__ 是类实例所属的类。于是我进行了一系列的实验: class counter: count = 0
也许这是一个微不足道的问题。在 python 下,为什么 self.__class__.__bases__ 没有回溯到 object?我的只显示了类(class)之上的单亲 parent 的元组。 最
我创建了这样的类来解决签名和 feature_names 的问题: import copy from sklearn.feature_selection import VarianceTreshold
我已阅读 Why is Python 3.x's super() magic?并了解在方法中使用 super 或 __class__ 会自动为该方法创建一个 __class__ 单元格变量: clas
我需要从类中获取类名: class Cls: notation = None def __init__(self): notation = self.__class__
我试图通过从该类派生并覆盖/添加某些方法来扩展用 C 编写的模块类的功能*。 问题是他的模块创建了我试图在不同地方扩展的类的实例。因此,我需要创建一个转换方法,将模块提供的基类转换为我自己的具有附加功
class Foo: pass >>> f = test.Foo() 让我们看看类实例...... >>> dir(f) ['__add__', [__class__] ...] 哦哦!让我们看看
我是一名优秀的程序员,十分优秀!