在 Python 中访问类的成员变量?
Posted
技术标签:
【中文标题】在 Python 中访问类的成员变量?【英文标题】:How do I access Class member variables in Python? 【发布时间】:2011-03-26 23:06:11 【问题描述】:class Example(object):
def the_example(self):
itsProblem = "problem"
theExample = Example()
print(theExample.itsProblem)
如何访问类的变量?我试过添加这个定义:
def return_itsProblem(self):
return itsProblem
然而,这也失败了。
【问题讨论】:
标题已编辑,实际上是关于类“静态”变量的问题:***.com/questions/707380/… 【参考方案1】:几句话就能给出答案
在您的示例中,itsProblem
是一个局部变量。
您必须使用self
来设置和获取实例变量。您可以在__init__
方法中设置它。那么您的代码将是:
class Example(object):
def __init__(self):
self.itsProblem = "problem"
theExample = Example()
print(theExample.itsProblem)
但是如果你想要一个真正的类变量,那么直接使用类名:
class Example(object):
itsProblem = "problem"
theExample = Example()
print(theExample.itsProblem)
print (Example.itsProblem)
但要小心这个,因为theExample.itsProblem
自动设置为等于Example.itsProblem
,但根本不是同一个变量,可以单独更改。
一些解释
在 Python 中,可以动态创建变量。因此,您可以执行以下操作:
class Example(object):
pass
Example.itsProblem = "problem"
e = Example()
e.itsSecondProblem = "problem"
print Example.itsProblem == e.itsSecondProblem
打印
是的
因此,这正是您对前面的示例所做的。
确实,在 Python 中,我们使用 self
和 this
,但不止于此。 self
是任何对象方法的第一个参数,因为第一个参数始终是对象引用。这是自动的,不管你是否称它为self
。
这意味着你可以做到:
class Example(object):
def __init__(self):
self.itsProblem = "problem"
theExample = Example()
print(theExample.itsProblem)
或:
class Example(object):
def __init__(my_super_self):
my_super_self.itsProblem = "problem"
theExample = Example()
print(theExample.itsProblem)
完全一样。 ANY 对象方法的第一个参数是当前对象,我们只是将其称为self
作为约定。 您只需向该对象添加一个变量,就像从外部执行此操作一样。
现在,关于类变量。
当你这样做时:
class Example(object):
itsProblem = "problem"
theExample = Example()
print(theExample.itsProblem)
您会注意到我们首先设置一个类变量,然后我们访问一个对象(实例)变量。我们从未设置过这个对象变量,但它可以工作,这怎么可能?
好吧,Python 会尝试首先获取对象变量,但如果找不到它,就会给你类变量。 警告:类变量是实例间共享的,而对象变量不是。
作为结论,永远不要使用类变量来为对象变量设置默认值。为此使用__init__
。
最终,您将了解到 Python 类是实例,因此也是对象本身,这为理解上述内容提供了新的见解。一旦你意识到这一点,请稍后再读一遍。
【讨论】:
您说:“theExample.itsProblem 自动设置为等于 Example.itsProblem,但根本不是同一个变量,可以独立更改”-但这不太正确,而且您的短语具有误导性。我相信你知道那里发生了什么,所以我建议重新表述:“它是同一个变量,但它可以为每个对象独立重新绑定”。 是的,但是绑定是一个完全未知的其他编程语言的概念,例如 Java 或 C(我怀疑 OP 是)。然后我会解释什么是绑定,然后是后期绑定,然后是可变对象上的引用问题。时间太长了。我认为有时你必须在理解的祭坛上牺牲精确性。 还有(我认为这可能是 2.7+)@classmethod 装饰器,值得研究。有趣的讨论在这里 - ***.com/questions/12179271/… name-binding:我认为给出虚假陈述是一个坏主意,无论是什么级别。我建议稍微修改一下:但要小心这个,因为theExample.itsProblem
自动设置为等于Example.itsProblem
,但从实际角度来看*,根本不是同一个变量,可以独立更改。 *:实际上它一开始是同一个对象,但如果你不了解 Python 的 name-binding,很容易意外更改它【参考方案2】:
您声明的是局部变量,而不是类变量。要设置实例变量(属性),请使用
class Example(object):
def the_example(self):
self.itsProblem = "problem" # <-- remember the 'self.'
theExample = Example()
theExample.the_example()
print(theExample.itsProblem)
要设置class variable(又名静态成员),请使用
class Example(object):
def the_example(self):
Example.itsProblem = "problem"
# or, type(self).itsProblem = "problem"
# depending what you want to do when the class is derived.
【讨论】:
除了类变量在class-level声明一次。你的方式会在每次实例化时重置它,这真的不是你想要的。另请参阅 ***.com/questions/2709821/python-self-explained 了解显式 self 背后的基本原理。【参考方案3】:如果您有一个实例函数(即传递 self 的实例函数),您可以使用 self 获取对使用 self.__class__
的类的引用
例如,在下面的代码中,tornado 创建了一个实例来处理 get 请求,但我们可以获取 get_handler
类并使用它来保存 riak 客户端,因此我们不需要为每个请求创建一个。
import tornado.web
import riak
class get_handler(tornado.web.requestHandler):
riak_client = None
def post(self):
cls = self.__class__
if cls.riak_client is None:
cls.riak_client = riak.RiakClient(pb_port=8087, protocol='pbc')
# Additional code to send response to the request ...
【讨论】:
这是一个很好的例子来展示来自 OPs 原标题的问题。实际上 OPs 示例与标题不匹配,其他答案参考示例。无论如何,这是访问类级别变量的最佳方式,因为它不违反 DRY 原则。就像其他一些例子一样。最好使用 self.__class__ 而不是重复类名。它使代码面向未来,使重构更容易,如果你敢于使用子类化,这也有好处。 应该警告读者,使用像上面这样的处理程序不仅会在非单线程应用程序中导致问题。理论上,类的多个实例可以并行使用相同的处理程序,如果处理程序不是为此而设计的,这取决于其内部结构,它可能无法工作。在单线程应用程序中“并行”意味着,第一个类实例尚未完成使用处理程序,然后第二个实例开始使用处理程序。在这种情况下,建议使用实例变量。【参考方案4】:像下面的例子那样实现 return 语句!你应该很好。我希望它可以帮助某人..
class Example(object):
def the_example(self):
itsProblem = "problem"
return itsProblem
theExample = Example()
print theExample.the_example()
【讨论】:
你应该修复你的代码缩进。这个问题也有更好的答案,您的答案是这些答案的基本变体,而不是替代解决方案......以上是关于在 Python 中访问类的成员变量?的主要内容,如果未能解决你的问题,请参考以下文章