1

我开始在 python 中创建类,我对这些术语感到非常困惑,所以我找不到我的问题的答案,因为我不知道如何问这些问题,因此这篇文章的双重标题。我将尝试使用以下代码澄清我不知道如何提出的问题:

我想在一个简单的类中创建一个方法:

class SimpleClass:
    add(self,Value)
        #Magic goes here

SC=SimpleClass()

该方法应该等效于向类中添加一个新变量:

SC.NewVariableInsideTheClass="test"

但是通过一种方法,例如:

SC.NewVariableInsideTheClass.add("test")

在这两种情况下,“NewVariableInsideTheClass”之前都不存在,我是第一次在类中创建该变量。

所以该方法应该首先创建不存在的实例(我认为这是类中变量的名称),然后将其值设置为“test”,但我真的找不到/搜索如何做到这一点.. .

哪个是我需要的正确问题?

非常感谢!

- - - - - - - - - 编辑 - - - - - - - - - - -

好的,所以我想做的是在执行以下操作时获得相同的结果:

SC.NewVariableInsideTheClass.add("test")

做的时候:

SC.NewVariableInsideTheClass.add("test")

但两者都必须在课外完成,第一次使用“NewVariableInsideTheClass”!这里的目标是使用类作为变量存储,比如当一个变量第一次被使用时会发生一些事情,如果它已经存在(以 SC.NewVariableInsideTheClass 的形式)我将能够以不同的方式使用它. 希望现在更具体...

第二次编辑;原因 - - - - - - - - - - - - -

哇,您的回答的质量和细节给我留下了深刻的印象,对此我非常感激!我想让你知道我想要这个的原因:我正在创建一个简单的 python 脚本 IDE(现在主要用于学习 python,nhot 还没有用)。因此,代码的基础是它会一路创建新变量。由于该程序还将创建函数,因此我创建和处理变量的第一个解决方案是将创建的程序中的所有变量设为全局变量,并在每个新函数中再次调用它们。那个代码很丑所以我决定学习和使用一个类,但是由于每个程序中的变量都不同,我想标准化函数的创建(即不需要知道它是否是第一个函数,如果它需要初始化任何变量,或者如果它们已经存在),能够:- 将新变量存储在名为变量的列表中,即使它不存在,其名称也可以轻松访问 - 能够使用新值扩展先前的变量作为列表中的新项目 - 能够检索那些values 全部不使用全局变量,我遇到了这个解决方案。实际上我修改了你的代码,现在我正在使用你的代码,但是我没有命名“add”,而是命名为“append”,所以我可以自由地做:

SC.NewVar.append('这是新值') SC.NewVar.append('这是新值')

第一次,它将被创建(因为它不存在)但第二次附加新值,因此,我将能够调用它/玩它或从自动生成的任何其他函数中使用它-代码 :)

4

2 回答 2

4

免责声明:我不知道你为什么要这样做。如果你想动态创建一个属性setattr应该足够了:

SC = SimpleClass()
setattr(SC, 'your_new_attribute_name', 'test')

无论如何,有一种简单的方法可以使用帮助类获得所需的内容:

In [1]: class SimpleClass:
   ...:     def __getattr__(self, attr):
   ...:         return AttributeSetter(self, attr)
   ...: class AttributeSetter:
   ...:     def __init__(self, inst, attr):
   ...:         self._inst = inst
   ...:         self._attr = attr
   ...:     def add(self, value):
   ...:         setattr(self._inst, self._attr, value)
   ...:         

In [2]: SC = SimpleClass()

In [3]: SC.new_attr.add('test')

In [4]: SC.new_attr
Out[4]: 'test'

关于你的评论。如果你问是否可以这样写SimpleClass

SC.new_attr

调用返回字符串 'test' and:_Sc.new_attr.add('test')

SC.new_attr.add('other')

是这样的:

SC.new_attr

现在返回字符串 'other',那么答案是:不,这是不可能的

编写一个类以使以下内容不产生是不可能AttributeError

In [2]: SC = SimpleClass()

In [3]: SC.new_attr.add('test')
In [4]: SC.new_attr   #this returns a string, not a user-defined class
'test'

In [5]: SC.new_attr.add('new value')
---------------------------------------------------------------------------
AttributeError                            Traceback (most recent call last)
<ipython-input-5-f218074b7a68> in <module>()
----> 1 SC.new_attr.add('new value')

AttributeError: 'str' object has no attribute 'add'

你为什么要问?好吧,让我们看看解释器实际上了什么:

In [5]: import dis
   ...: dis.dis(lambda: SC.new_attr.add('some string'))
   ...: 
  2           0 LOAD_GLOBAL              0 (SC) 
              3 LOAD_ATTR                1 (new_attr) 
              6 LOAD_ATTR                2 (add) 
              9 LOAD_CONST               1 ('some string') 
             12 CALL_FUNCTION            1 (1 positional, 0 keyword pair) 
             15 RETURN_VALUE 

dis模块允许我们查看解释器正在执行的字节码。请注意如何有两个不同的 字节LOAD_ATTR码。如您所见,解释器首先获取new_attr属性,然后作为完全独立的操作,它检索add属性。

现在,该类SC只能修改第一个属性访问(即对 的访问new_attr)的行为。它无法知道下一个字节码将访问add,因此唯一明智的做法是返回设置的字符串。

对检索到的值add进行以下属性访问。这意味着解释器会寻找字符串的属性,这显然不存在。add

避免这种AttributeError情况的唯一方法是返回一个具有add方法的对象,但这意味着这样做:

In [6]: SC.new_attr
Out[6]: 'test'

失败,因为我们不会返回字符串,"test"而是返回一个不同的对象。

解决这个问题的一种方法是返回一个具有add方法和value属性的对象,这样我们就可以获得类似的东西:

In [12]: SC.new_attr
Out[12]: <__main__.TheClass at 0x7fac44727a90>

In [13]: SC.new_attr.value
Out[13]: 'test'

In [14]: SC.new_attr.add('other')

In [15]: SC.new_attr.value
Out[15]: 'other'

一个不同的解决方案是编写这个类,使其表现得像一个字符串(我们甚至可以子类化str不要这样做!内置子类化永远不是正确的做法,如果你在不知道确切后果的情况下这样做您遇到问题,并且很难发现和解决),但这意味着代码将变得更加复杂,必须实现字符串支持的所有操作(这意味着实现大约 80-90-100 个方法!)

于 2013-07-16T09:17:09.730 回答
0

让我们用方法 add 声明一个类 ABC

class ABC(object):
    def add(self, var):
         self.new_var = var

现在 add 是实例方法,从而完成了您要做的事情,即在实例方法中声明一个实例变量。

 a = ABC()
 # Make an instance
 a.add('test')
 # created a new variable inside instance
 a.new_var
 'test'
于 2013-07-16T09:11:05.253 回答