Python封装机制及实现方法

不光是Python,大多数面向对象编程语言(诸如C++、Java等)都具备3个典型特征,即封装、继承和多态。其中,本节重点讲解Python类的封装特性,继承和多态会在后续章节给大家做详细讲解。

简单的理解封装(Encapsulation),即在设计类时,刻意地将一些属性和方法隐藏在类的内部,这样在使用此类时,将无法直接以“类对象.属性名”(或者“类对象.方法名(参数)”)的形式调用这些属性(或方法),而只能用未隐藏的类方法间接操作这些隐藏的属性和方法。

就好比使用电脑,我们只需要学会如何使用键盘和鼠标就可以了,不用关心内部是怎么实现的,因为那是生产和设计人员该操心的。

注意,封装绝不是将类中所有的方法都隐藏起来,一定要留一些像键盘、鼠标这样可供外界使用的类方法。

那么,类为什么要进行封装,这样做有什么好处呢?

首先,封装机制保证了类内部数据结构的完整性,因为使用类的用户无法直接看到类中的数据结构,只能使用类允许公开的数据,很好地避免了外部对内部数据的影响,提高了程序的可维护性。

除此之外,对一个类实现良好的封装,用户只能借助暴露出来的类方法来访问数据,我们只需要在这些暴露的方法中加入适当的控制逻辑,即可轻松实现用户对类中属性或方法的不合理操作。

并且,对类进行良好的封装,还可以提高代码的复用性。

Python类如何进行封装?

和其它面向对象的编程语言(如C++、Java)不同,Python类中的变量和函数,不是公有的(类似public属性),就是私有的(类似private),这2种属性的区别如下:

public:公有属性的类变量和类函数,在类的外部、类内部以及子类(后续讲继承特性时会做详细介绍)中,都可以正常访问;

private:私有属性的类变量和类函数,只能在本类内部使用,类的外部以及子类都无法使用。

但是,Python并没有提供public、private这些修饰符。为了实现类的封装,Python采取了下面的方法:

默认情况下,Python类中的变量和方法都是公有(public)的,它们的名称前都没有下划线(_);

如果类中的变量和函数,其名称以双下划线“__”开头,则该变量(函数)为私有变量(私有函数),其属性等同于private。

除此之外,还可以定义以单下划线“_”开头的类属性或者类方法(例如_name、_display(self)),这种类属性和类方法通常被视为私有属性和私有方法,虽然它们也能通过类对象正常访问,但这是一种约定俗称的用法,初学者一定要遵守。

注意,Python类中还有以双下划线开头和结尾的类方法(例如类的构造函数__init__(self)),这些都是Python内部定义的,用于Python内部调用。我们自己定义类属性或者类方法时,不要使用这种格式。

例如,如下程序示范了Python的封装机制:

class CLanguage :
 def setname(self, name):
  if len(name) < 3:   raise ValueError('名称长度必须大于3!')
  self.__name = name def getname(self):
  return self.__name #为 name 配置 setter 和 getter 方法
 name = property(getname, setname) def setadd(self, add):
  if add.startswith("http://"):
   self.__add = add  else:   raise ValueError('地址必须以 http:// 开头')
 def getadd(self):
  return self.__add
 #为 add 配置 setter 和 getter 方法
 add = property(getadd, setadd) #定义个私有方法
 def __display(self):
  print(self.__name,self.__add)
clang = CLanguage()
clang.name = "开课吧广场"clang.add = "https://topic.kaikeba.com/"print(clang.name)
print(clang.add)123456789101112131415161718192021222324252627复制代码类型:[python]

程序运行结果为:

开课吧广场
https://topic.kaikeba.com/12复制代码类型:[python]

上面程序中,CLanguage将name和add属性都隐藏了起来,但同时也提供了可操作它们的“窗口”,也就是各自的setter和getter方法,这些方法都是公有(public)的。

不仅如此,以add属性的setadd()方法为例,通过在该方法内部添加控制逻辑,即通过调用startswith()方法,控制用户输入的地址必须以“http://”开头,否则程序将会执行raise语句抛出ValueError异常。

有关raise的具体用法,后续章节会做详细的讲解,这里可简单理解成,如果用户输入不规范,程序将会报错。

通过此程序的运行逻辑不难看出,通过对CLanguage类进行良好的封装,使得用户仅能通过暴露的setter()和getter()方法操作name和add属性,而通过对setname()和setadd()方法进行适当的设计,可以避免用户对类中属性的不合理操作,从而提高了类的可维护性和安全性。

细心的读者可能还发现,CLanguage类中还有一个__display()方法,由于该类方法为私有(private)方法,且该类没有提供操作该私有方法的“窗口”,因此我们无法在类的外部使用它。换句话说,如下调用__display()方法是不可行的:

#尝试调用私有的 display() 方法clang.__display()12复制代码类型:[python]

这会导致如下错误:

Traceback (most recent call last):
  File "D:\python3.6\1.py", line 33, in <module>
 clang.__display()
AttributeError: 'CLanguage' object has no attribute '__display'
(0)

相关推荐

  • 类属性与方法(私有)

    类的方法 在类的内部,使用 def 关键字定义方法,与一般函数定义不同,类方法必须包含参数 self,且为第一个参数 类的私有方法 __private:两个下划线开头声明的方法为私有方法,不能在类地外 ...

  • Python|二叉树叶子结点问题解决方法

    问题描述键盘输入一颗二叉树,求解其叶子结点个数.示例: 输入:4,2,6,1,3,5输出:3解决方案一棵树当中没有子结点(即度为0)的结点称为叶子结点,简称"叶子".当二叉树为空时 ...

  • Python|二叉树的遍历问题解决方法

    问题描述二叉树是由n个结点的有限集合,该集合或者为空集,或者由一个根节点和两颗互不相交的.分别称为根节点的左子树和右子树的二叉树组成.二叉树特征:每个结点最多只有两颗子树,即二叉树中结点的度最高不能超 ...

  • Python优化机制:常量折叠

    英文:https://arpitbhayani.me/blogs/constant-folding-python 作者:arprit 译者:豌豆花下猫("Python猫"公众号作者 ...

  • PD-1/L1或CTLA-4抗体治疗引发的炎症毒性机制与应对方法

    PD-1/L1或CTLA-4这类检查点抑制剂的抗体至今临床已15年有余,伴随其较好的抗肿瘤作用,其炎症毒性也是临床用药的关注重点.炎症症状大多在治疗后的几个月内就能产生,可以发生在单独器官,也可以是系 ...

  • Selenium2+python自动化45-18种定位方法(find_elements)

    前言 江湖传言,武林中流传八种定位,其中xpath是宝刀屠龙,css是倚天剑. 除了这八种,其实还有十种定位方法,眼看就快失传了,今天小编让失传已久的定位方法重出江湖! 一.十八种定位方法 前八种是大 ...

  • Common encryption methods and implementation in Python Python中常用的加密方法及实现

    Common encryption methods and implementation in Python Python中常用的加密方法及实现

  • Python 下载文件的多种方法

    来源:Python 技术「ID: pythonall」 本文档介绍了 Python 下载文件的各种方式,从下载简单的小文件到用断点续传的方式下载大文件. Requests 使用 Requests 模块 ...

  • python 包机制、导入模块

    表白:黑白圣堂血天使,天剑鬼刀阿修罗.  讲解对象:/python 包机制.导入模块 作者:融水公子 rsgz Python3 教程 python教程 http://www.rsgz.top/post ...

  • 列表推导式:简洁高效更具 Python 风格的列表创建方法

    我们在<Python 中的列表和元组>中已经详细介绍了列表(list)的基本特性和使用方法,本文将着重介绍一种 Python 中用于创建 list 的简洁高效的语法形式:列表推导式. Py ...