美文网首页
Python @property

Python @property

作者: Robots小助手 | 来源:发表于2020-02-01 11:40 被阅读0次

    以下内容来自万能的互联网...

    首先教材上的
    @property 可以将一个方法的调用变成“属性调用”,主要用于帮助我们处理属性的都操作和写操作,对于某一个对象,可以直接通过:a.salary = 3000。

    下面是网上的,讲的比较详细了,如果用纸画画内存图可能会更清晰。

    在绑定属性时,如果我们直接把属性赋值给对象,比如:

    p = Person()
    p.name= 'Mary'
    

    我们先看个详细的例子(注意双下划线name和age定义为私有变量):

    class Person(object):
        def __init__(self, name, age):
            self.__name = name
            self.__age = age
    
        def get_age_fun(self):
             return self.__age
    
        def set_age_fun(self, value):
            if not isinstance(value, int):
                raise ValueError('年龄必须是数字!')
            if value < 0 or value > 100:
                raise ValueError('年龄必须是0-100')
            self.__age = value
    
        def print_info(self):
            print('%s: %s' % (self.__name, self.__age))
    p = Person('balala',20)
    p.__age = 17
    print(p.__age) # 17
    print(p.get_age_fun()) # 20 
    

    表面上看,上面代码“成功”地设置了__age变量 17,但实际上这个__age变量和class内部的__age变量不是一个变量! 内部的__age变量已经被Python解释器自动改成了_Person__age,而外部代码给p新增了一个__age变量。 所以调用 get_age_fun输出的是初始值

    p.set_age_fun(35)
    print(p.get_age_fun()) # 35
    print(p.print_info()) # balala: 35
    

    我们再稍微调整下:
    (注意只改变了一个变量名: 原来的私有属性 __age 单下划线为: _age,也可以定义为:age. 解释:以一个下划线开头的实例变量名,比如_age,这样的实例变量外部是可以访问的,但是,按照约定俗成的规定,当看到这样的变量时,意思是,"虽然可以被访问,但是,请视为私有变量,不要随意访问。")

    class Person(object):
        def __init__(self, name, age):
            self.__name = name
            self._age = age
    
        def get_age_fun(self):
             return self._age
    
        def set_age_fun(self, value):
            if not isinstance(value, int):
                raise ValueError('年龄必须是数字!')
            if value < 0 or value > 100:
                raise ValueError('年龄必须是0-100')
            self._age = value
    
        def print_info(self):
            print('%s: %s' % (self.__name, self._age))
    
    
    p = Person('balala',20)
    p._age = 17
    print(p._age) # 17
    print(p.get_age_fun()) # 这里是17 不再是 20,因为此时_age是全局变量,外部直接影响到类内部的更新值
    p.set_age_fun(35)
    print(p.get_age_fun()) # 35
    print(p.print_info()) # balala: 35
    

    看的出私有和全局的设置,但是,上面的调用方法是不是略显复杂,没有直接用属性这么直接简单。有没有可以用类似属性这样简单的方式来访问类的变量呢?必须的,对于类的方法我们先来看一个稍微改造的例子:
    (稍后我们再使用Python内置的@property装饰器就是负责把一个方法变成属性调用.)
    我们进入正题:看看@property的妙用之处:

    class Person(object):
        def __init__(self, name, age):
            self.__name = name
            self.__age = age
    
        @property
        def get_age_fun(self):
             return self.__age
    
        @get_age_fun.setter # get_age_fun是上面声明的方法
        def set_age_fun(self, value):
            if not isinstance(value, int):
                raise ValueError('年龄必须是数字!')
            if value < 0 or value > 100:
                raise ValueError('年龄必须是0-100')
            self.__age = value
    
        def print_info(self):
            print('%s: %s' % (self.__name, self.__age))
    
    
    p = Person('balala',20)
    p.__age = 17
    print(p.__age) # 17
    print(p.get_age_fun) # 20 注意这里不带()
    
    #p.set_age_fun(35) 注意不能这样调用赋值了
    p.set_age_fun = 35 #  这里set_age_fun 就是 声明的函数不带()
    print(p.get_age_fun) # 35
    print(p.print_info()) # balala: 35
    

    参考文章

    https://www.cnblogs.com/phpper/p/10618775.html

    相关文章

      网友评论

          本文标题:Python @property

          本文链接:https://www.haomeiwen.com/subject/xxgtxhtx.html