在 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 中,我们使用 selfthis,但不止于此。 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 中访问类的成员变量?的主要内容,如果未能解决你的问题,请参考以下文章

理解java和python类变量以及类的成员变量

Java继承

继承中成员的访问特点

让公共成员变量访问 C++ 中同一类的私有成员

MFC中静态成员函数调用其他类的非静态变量

C++ 使用变量访问类的公共成员