Python核心编程笔记--私有化

一、私有化的实现

在Python中想定义一个类是比较简单的,比如要定义一个Person类,如下代码即可:

# -*- coding: utf-8 -*-
# __author : Demon
# date : 1/5/18 8:24 PM

class Person(object):
    pass

p1 = Person()

当然我们也可以给类添加相应的属性,比如Person的姓名,年龄,性别等,并且在new一个Person对象后可以对这些属性进行修改,如下代码所示:

# -*- coding: utf-8 -*-
# __author : Demon
# date : 1/5/18 8:24 PM

class Person(object):
    def __init__(self, name, age, sex):
        self.name = name #姓名
        self.age = age #年龄
        self.sex = sex #性别

    def __str__(self): #定义直接打印Person对象,输出的内容。相当于Java中的toString()方法
        return ('%s , %s , %s '  % (self.name, self.age, self.sex))

p1 = Person('Demon', 18, 'M')
print(p1) #Demon , 18 , M 

p1.age = 28 #修改对象的年龄
print(p1) #Demon , 28 , M 

Python是面向对象的语言,我们都知道面向对象的三大特点:继承、封装和多态。而在上面的代码中,我们可以轻易地通过person. 调用属性来访问和修改值,这明显不符合面向对象中封装的思想。比如对于Person类中的年龄属性,可能比较私人,所以我们不希望可以任意的访问和修改它的值。在Java中,我们是通过private关键字来装饰属性私有,那么在Python中有没有类似的关键字呢?显然是有的,在Python中采取在变量名前加__(两个下划线)的方式来将属性私有。如下代码所示:

# -*- coding: utf-8 -*-
# __author : Demon
# date : 1/5/18 8:24 PM

class Person(object):
    def __init__(self, name, age, sex):
        self.name = name #姓名
        self.__age = age #年龄,将年龄私有
        self.sex = sex #性别

    def __str__(self): #定义直接打印Person对象,输出的内容。相当于Java中的toString()方法
        return ('%s , %s , %s '  % (self.name, self.__age, self.sex))

p1 = Person('Demon', 18, 'M')
print(p1) #Demon , 18 , M

p1.age = 28 #这里只是给p1对象动态的增加了一个age属性
print(p1) #Demon , 18 , M
print(p1.age) #28

初看上面的代码,似乎感觉并没有达到私有的目的,因为我们通过p1.age还是成功地进行了赋值,程序并没有报错。其实这里是因为Python可以动态地给对象增加属性和方法,这句话相当于动态地给p1这个对象增加了一个age属性。所以在我们做了赋值操作之后,我们再次打印p1,age的值依然是18。说明我们确实已经将类里的age属性进行的私有。同理方法的私有也是一样的处理。

 

二、私有化的注意事项

在Python中,有几种方式来定义变量:

1、以单划线开头:这种类型的变量可以通过对象.调用,但是它表示的意思是我可以调用,但请把我视为是私有的。而且如果是通过from xxx_module import *是无法访问的,但是如果是import xxx_module的方式,则可以访问到,类对象和子类也都可以访问。

2、仅以双划线开头:这种类型的变量就是私有。但是它可以通过_类名__变量名来访问,但强烈建议不要这样做

3、以双划线开头,并以它结尾:这种类型的变量在Python中通常都表示具有特殊意义的变量,比如__init__,__str__等。所以我们在定义变量时不要这样定义

4、仅以单划线结尾:这种类型的变量是用于避免与Python关键字进行冲突所采取的一种解决办法

如下图演示访问权限效果:

 

三、property的使用

上面的介绍了,通过加双划线开头的方式实现了变量和方法。参照在学Java时的思路,如果想要访问私有变量,我们会给变量添加get和set方法。同样我们在Python中也是一样的处理,代码如下所示:

class Person(object):
    def __init__(self, name, age, sex):
        self.name = name #姓名
        self.__age = age #年龄,将年龄私有
        self.sex = sex #性别

    def getAge(self):
        return self.__age

    def setAge(self, age):
        self.__age = age

p1 = Person('Demon', 18, 'M')
print(p1.getAge()) #18

p1.setAge(28)
print(p1.getAge()) #28

但是这样看着似乎不是很方便,每次都要调用一个方法。有没有可能像之前一样调用p1. age = 28就能直接赋值呢?显然在Python中是可以的,这就要用到property。

property的定义:

使用代码示例:

class Person(object):
    def __init__(self, name, age, sex):
        self.name = name #姓名
        self.__age = age #年龄,将年龄私有
        self.sex = sex #性别

    def getAge(self):
        return self.__age

    def setAge(self, age):
        self.__age = age

    age = property(getAge, setAge, 'This is age property')

p1 = Person('Demon', 18, 'M')
print(p1.age) #18

p1.age = 28
print(p1.age) #28

说明:

1. property接受四个参数,分别是get, set, del, doc,前三个参数分别对应get方法,set方法,del方法,顺序不能出错。最后一个参数是doc,相当于对方法进行说明。

2. property返回一个property属性,返回值的变量名与最终对象. 后面的名称是一致的

 

观察上面的代码,我们仍然需要多写一行property的代码,而Python其实提供了一个更方便的实现方式来达到上述要求,即使用@property。说明如下:

示例代码如下: 

class Person(object):
    def __init__(self, name, age, sex):
        self.name = name #姓名
        self.__age = age #年龄,将年龄私有
        self.sex = sex #性别

    @property
    def age(self):  #注意方法名直接为变量名
        return self.__age

    @age.setter     #注意方法名直接为变量名
    def age(self, age):
        self.__age = age


p1 = Person('Demon', 18, 'M')
print(p1.age) #18

p1.age = 28
print(p1.age) #28

 说明:

1. @property相当于对age方法进行了一个装饰,它使得我们能通过对象.方法名来调用对应的属性

2. @property所装饰的方法名与对象 . 调用的名称要保持一致

3. @property会生成另外的装饰器,@方法名.setter, @方法名.getter, @方法名.deleter,分别对应set, get, del方法。这里get方法用得很少,因为已经通过@property直接对应到了get方法 

 

 

 

posted on 2018-01-08 10:52  DemonFS  阅读(613)  评论(0编辑  收藏  举报

导航