本文首发自「慕课网」,想了解更多IT干货内容,程序员圈内热闻,欢迎关注!

作者| 慕课网精英讲师 朱广蔚

在 Python 的面向对象编程中,私有属性是只能在类的实例方法中访问的属性,不允许在外界访问私有属性。

  1. 私有属性的定义
    1.1 定义
    在属性名称前加上前缀 __,表示该属性为私有属性,示例代码如下:

class Object:

def method(self):
    self.__private_attribute = 123

代码块123
在第 3 行,创建一个私有属性 __private_attribute。

1.2 在类外读取私有属性
只能在类的实例方法中访问私有属性,不允许在类的外部访问私有属性,示例代码如下:

class Person:

def __init__(self, name):
    self.__name = name

tom = Person('tom')
print(tom.__name)
代码块123456
在第 1 行,定义了类 Person
在第 3 行,创建私有属性 __name
在第 5 行,创建一个实例 tom
在第 6 行,直接访问实例的属性 __name
程序运行输出如下:

Traceback (most recent call last):
File "attr-get.py", line 6, in <module>

print(tom.__name)

AttributeError: 'Person' object has no attribute '__name'
代码块1234
程序运行报错:‘Person’ object has no attribute ‘__name’,表示无法找到属性 __name’。因此,在类 Person 的外部无法直接读取实例的私有属性。

1.3 在类外修改私有属性
1.2 小节的例子,在类外读取私有属性;本节的例子,在类外修改私有属性。示例代码如下:

class Person:

def __init__(self, name):
    self.__name = name

def get_name(self):
    return self.__name

tom = Person('tom')
tom.__name = 'jerry'
print(tom.get_name())
代码块12345678910
在第 1 行,定义了类 Person在第 3 行,创建私有属性 __name在第 6 行,在类的实例方法 get_name 中,访问私有属性 __name
在第 8 行,创建一个实例 tom在第 9 行,将实例的属性 __name 修改为 ‘jerry’在第 10 行,通过实例方法 get_name 读取私有属性 __name
程序运行输出如下:

tom
代码块1
程序在第 9 行,将实例的私有属性 __name 修改为 ‘jerry’,但是程序输出表明:在类的内部,私有属性 __name 没有发生变化。因此,在类 Person 的外部无法直接修改实例的私有属性。

1.4 通过 set/get 方法访问私有属性
本节在类的外部访问私有属性的方法,代码如下:

class Person:

def __init__(self, name):
    self.__name = name

def get_name(self):
    return self.__name

def set_name(self, name):
    self.__name = name

tom = Person('tom')
tom.set_name('jerry')
print(tom.get_name())
代码块12345678910111213
在第 1 行,定义了类 Person在第 3 行,创建私有属性 __name在第 5 行,创建方法 get_name,它读取私有属性 __name在第 8 行,创建方法 set_name,它修改私有属性 __name
在第 11 行,创建一个实例 tom在第 12 行,通过实例方法 get_name 读取私有属性 __name在第 13 行,通过实例方法 set_name 修改私有属性 __name
程序输出结果如下:

jerry
代码块1
程序输出表明,通过方法 tom.set_name(‘jerry’) 成功的将私有属性 __name 设置为 jerry。因此,在类的外部通过 get/set 方法访问私有属性。

  1. 私有属性的应用
    2.1 概述
    数学中的线段拥有 3 个属性:

start,表示开始位置
end,表示结束位置
length,表示线段的长度,等于 end - start
当修改属性 start 时,属性 length 会发生变化;当修改属性 end 时,属性 length 也会发生变化;如果修改属性 start 或者 end 时,忘记修改属性 length 的话,则会造成逻辑错误,示例代码如下:

class Segment:

def __init__(self, start, end):
    self.start = start
    self.end = end
    self.length = self.end - self.start

def show(self):
    print('start = %d, end = %d, length = %d' % (self.start, self.end, self.length))

segment = Segment(10, 100)
segment.start = 20
segment.show()
代码块123456789101112
在第 2 行,定义构造方法在第 5 行,使用属性 start 和 end 计算属性 length
在第 7 行,定义方法 show,打印属性 start、end、length
在第 10 行,创建线段 segment,设置 start = 10,end = 100
在第 11 行,将 start 修改为 20
在第 12 行,调用方法 show 打印属性 start、end、length
程序运行输出结果如下:

start = 20, end = 100, length = 90
代码块1
start 修改为 20 后,length 应该等于 80,但是程序输出表明 length 等于 90。由于 start 修改后,忘记修改 length,造成了这样的逻辑错误。

2.2 使用私有属性解决问题
为了解决上个小节中的问题,将属性 start、end、length 设置为私有属性:

禁止在外界直接访问这 3 个属性
只能通过对应的 get/set 方法访问这 3 个属性
class Segment:

def __init__(self, start, end):
    self.__start = start
    self.__end = end
    self.__length = self.__end - self.__start

def get_start(self):
    return self.__start

def set_start(self, start):
    self.__start = start
    self.__length = self.__end - self.__start

def get_end(self):
    return self.__end

def set_end(self, end):
    self.__end = end
    self.__length = self.__end - self.__start

def get_length(self):
    return self.__start

def set_length(self, length):
    self.__length = length

def show(self):
    print('start = %d, end = %d, length = %d' % (self.__start, self.__end, self.__length))

segment = Segment(10, 100)
segment.set_start(20)
segment.show()
代码块1234567891011121314151617181920212223242526272829303132
类 Segment,包含 3 个私有属性,读写这些属性的方法如下:

方法

功能

get_start

读取属性 start

set_start

设置属性 start

get_end

读取属性 end

set_end

设置属性 end

get_length

读取属性 length

set_length

设置属性 length

在第 12 行,在 set_start 方法中,修改属性 start 时,同时重新计算属性 length
在第 19 行,在 set_end 方法中,修改属性 end 时,同时重新计算属性 length
在 set 方法中,修改 start 和 end 属性时,同时修改 length 属性,从而保证了一致性,不会出现上个小节中的逻辑错误。
程序运行输出结果如下:

start = 20, end = 100, length = 80
代码块1
输出表明,当属性 start 修改为 20 后,属性 length 被修改为 80,避免了上个小节中的错误。

  1. 私有方法的定义
    3.1 定义
    在Python 的面向对象编程中, 私有方法是只能在类的实例方法中访问的方法,不允许在外界访问私有方法。在方法名称前加上前缀 __,表示该方法为私有方法,示例代码如下:

class Object:

def __private_method(self):
    pass

代码块123
在第 3 行,定义了一个私有方法 __private_method。

3.2 在类外访问私有方法
私有方法只能在类的内部被调用,不允许在类的外部访问。示例代码如下:

class Object:

def __private_method(self):
    pass

object = Object()
object.__private_method()
代码块123456
在第 2 行,定义了一个私有方法 __private_method
在第 5 行,创建一个实例 object
在第 6 行,调用实例的私有方法 __private_method
程序运行输出如下:

Traceback (most recent call last):
File "method-error.py", line 6, in <module>

object.__private_method()

AttributeError: 'Object' object has no attribute '__private_method'
代码块1234
程序运行报错:‘Object’ object has no attribute ‘__private_method’,表示无法找到方法 __private_method’。因此,在类 Person 的外部无法调用实例的私有方法。

  1. 私有方法的应用
    4.1 概述
    本节完成一个分析文本的程序,文本由多个单词构成,单词之间使用空格隔开,单词的类型如下:

数字,例如 123
字母,例如 abc
操作符,例如 =、+、- 等符号
程序对文本分析后,输出单词的类型和值,假设输入文本为 a = 123,则输出如下:

alpha a
operator =
digit 123
代码块123
程序的思路如下:

定义方法 parse_string,它将文本分割为多个单词
定义方法 parse_word,它判断并打印单词的类型
在方法 parse_string 中调用 parse_word 处理每个单词
方法 parse_word 用于辅助实现方法 parse_string,不需要被外界访问,因此将其设定为私有方法。

4.2 使用私有方法解决问题
class Parser:

def __parse_word(self, word):
    if word.isdigit():
        print('digit    %s' % word)
    elif word.isalpha():
        print('alpha    %s' % word)
    elif word == '=' or word == '+' or word == '-':
        print('operator %s' % word)

def parse_string(self, string):
    words = string.split(' ')
    for word in words:
        self.__parse_word(word)

parser = Parser()
parser.parse_string('sum = sum + 100')
代码块12345678910111213141516
在第 2 行,定义私有方法 __parse_word,判断单词的类型在第 3 行,通过方法 isdigit 判断是否为数字在第 5 行, 通过方法 isalpha 判断是否为字母
在第 10 行,定义公开方法 parse_string在第 11 行,使用 split 将文本分割为多个单词在第 13 行,循环调用私有方法 __parse_word 处理每个单词
在第 16 行,在类 Parser 的外部,调用公开方法 parse_string
实现方法 parse_string 是类 Parser 的接口,外界通过这个方法实现分析文本的功能;而方法 __parse_word 是一个辅助方法,它用于实现方法 parse_string,不需要公开给外界调用,因此将它设定为私有方法。

程序运行输出如下:

alpha sum
operator =
alpha sum
operator +
digit 100
代码块12345

欢迎关注「慕课网」,发现更多IT圈优质内容,分享干货知识,帮助你成为更好的程序员!


慕课网
29 声望8 粉丝

慕课网,国内深受欢迎的互联网IT技能学习网站,IT教育行业的造梦者,也是前沿技术内容的创造者和传播者!在慕课网,只学有用的。