search

Home  >  Q&A  >  body text

关于下面这段python代码的一点疑问?

这段代码是《python核心编程》中的例子,是讲property()函数的。
代码如下:

class HideX(object):
    def __init__(self, x):
        self.x = x
    def get_x(self):
        return ~self.__x
    def set_x(self, x):
        assert isinstance(x, int), '"x" must be an integer!'
        self.__x = ~x
    x = property(get_x, set_x)

在__init__中看到实例属性叫作x,而在get_x和set_x中实例属性却用的__x。我运行了一下,结果没问题。
结果如下图:

我想问,在__init__与get_x、set_x用的属性名不一样,为什么结果不会出错?在一个变量名前加上__不是意味着就成了另一个变量名了?

大家讲道理大家讲道理2892 days ago381

reply all(2)I'll reply

  • 黄舟

    黄舟2017-04-18 09:08:38

    #我们首先来证明一下,self.x和self.__x不是同一个变量
    class HideX(object):
        def __init__(self, x):
            self.x = x
            self.__x = x + 1
    
    inst = HideX(1)
    print(inst.x)#输出 1
    print(inst._HideX__x)#输出 2 
    #为什么用_HideX__x,可以参考有关python下划线相关的文章 http://blog.csdn.net/pfm685757/article/details/45918575
    
    #其次,我们来证明一下,self.x的值变化与self.__x变量名取名无关
    
    class HideX(object):
        def __init__(self, x):
            self.x = x
        def get_x(self):
            return ~self.__y #改了这个
        def set_x(self, x):
            assert isinstance(x, int), '"x" must be an integer!'
            self.__y = ~x #改了这个
        x = property(get_x, set_x)
    
    inst = HideX(0)
    print(inst.x) # 0
    inst.x = -1
    print(inst.x) # -1
    
    #最后,我们来证明,之前的self.x是property,而不是一般的实例变量
    class HideX(object):
        def __init__(self, x):
            self.x = x
        def get_x(self):
            return 100 #改了这个
        def set_x(self, x):
            assert isinstance(x, int), '"x" must be an integer!'
            self.__x = ~x
        x = property(get_x, set_x)
    
    inst = HideX(0)
    print(inst.x) #100
    
    #若是上面还不能明显说服,那再将x变成y
    
    
    class HideX(object):
        def __init__(self, x):
            self.x = x
        def get_x(self):
            return 100 #改了这个
        def set_x(self, x):
            assert isinstance(x, int), '"x" must be an integer!'
            self.__x = ~x
        y = property(get_x, set_x) #改了这个
    
    inst = HideX(0)
    print(inst.x) # 0
    print(inst.y) # 100
    
    #接下来,回答你的问题
    #在__init__与get_x、set_x用的属性名不一样,为什么结果不会出错?
    #因为并没有语法错误,get_x调用的__x在set_x中赋值了
    #在一个变量名前加上__不是意味着就成了另一个变量名了
    #是的

    reply
    0
  • 伊谢尔伦

    伊谢尔伦2017-04-18 09:08:38

    def __init__(self, x):
      self.x = x

    x here is not a variable
    but a property
    will be called set_x

    reply
    0
  • Cancelreply