Python从菜鸟到高手(18):类与方法的私有化

1. 创建自己的类

学习面向对象的第一步,就是创建一个类。因为类是面向对象的基石。Python类和其他编程语言(Java、C#等)的类差不多,也需要使用class关键字。下面通过一个实际的例子来看一下Python类是如何创建的。

本例会创建一个类,以及利用这个类创建两个对象,并调用其中的方法。

 1 # 创建一个Person类
 2 class Person:
 3     # 定义setName方法
 4     def setName(self, name):
 5         self.name = name
 6     # 定义getName方法
 7     def getName(self):
 8         return self.name
 9     # 定义greet方法
10     def greet(self):
11         print("Hello, I‘m {name}.".format(name = self.name))
12
13 # 创建Person对象
14 person1 = Person()
15 # 创建Person对象
16 person2 = Person()
17 # 调用person1对象的setName方法
18 person1.setName("Bill Gates")
19 # 调用person2对象的name属性
20 person2.name = "Bill Clinton"
21 # 调用person1对象的getName方法
22 print(person1.getName())
23 # 调用person1对象的greet方法
24 person1.greet()
25 # 调用person2对象的属性
26 print(person2.name)
27 # 调用person2对象的greet方法,另外一种调用方法的方式
28 Person.greet(person2)

程序运行结果如下图所示。

从上面的代码我们可以了解到Python类的如下知识点。

  • Python类使用class关键字定义,类名直接跟在class关键字的后面。
  • 类也是一个代码块,所以类名后面要跟着一个冒号(:)。
  • 类中的方法其实就是函数,定义的方法也完全一样,只是由于函数定义在类的内部,所以为了区分,将定义在类内部的函数称为方法。
  • 我们可以看到,每一个方法的第1个参数都是self,其实这是必须的。这个参数名不一定叫self(可以叫abc或任何其他名字),但任意一个方法必须至少指定一个self参数,如果方法中包含多个参数,第1个参数将作为self参数使用。在调用方法时,这个参数的值不需要自己传递,系统会将方法所属的对象传入这个参数。在方法内部可以利用这个参数调用对象本身的资源,如属性、方法等。
  • 通过self参数添加的name变量是Person类的属性,可以在外部访问。本例设置了person2对象的name属性的值,与调用person2.setName方法的效果完全相同。
  • 使用类创建对象的方式与调用函数的方式相同。在Python语言中,不需要像Java一样使用new关键字创建对象,只需要用类名加上构造方法(在后面的章节会详细介绍)参数值即可。
  • 调用对象的方法有两种方式,一种是直接通过对象变量调用方法,另一种是通过类调用方法,并且将相应的对象传入方法的第1个参数。在本例中使用了Person.greet(person2)的方式调用了person2对象中的greet方法。

如果使用集成开发环境,如PyDev、PyCharm,那么代码编辑器也会对面向对象有很好的支持,例如,当在对象变量后输入一个点(.)后,IDE会为我们列出该对象中所有可以调用的资源,包括方法和属性,如下图所示。

2.方法和私有化

Python类默认情况下,所有的方法都可以被外部访问。不过像很多其他编程语言,如Java、C#等,都提供了private关键字将方法私有化,也就是说只有类的内部方法才能访问私有化的方法,通过正常的方式是无法访问对象的私有化方法的(除非使用反射技术,这就另当别论了)。不过在Python类中并没有提供private或类似的关键字将方法私有化,但可以曲线救国。

在Python类的方法名前面加双下划线(__)可以让该方法在外部不可访问。

 1 class Person:
 2     # method1方法在类的外部可以访问
 3     def method1(self):
 4         print("method1")
 5     # __method2方法在类的外部不可访问
 6     def __method2(self):
 7         print("method2")
 8
 9 p = Person()
10 p.method1()
11 p.__method2()       # 抛出异常

如果执行上面的代码,会抛出如下图所示的异常信息,原因是调用了私有化方法method2。

其实“method2”方法也不是绝对不可访问。Python编译器在编译Python源代码时并没有将“method2”方法真正私有化,而是一旦遇到方法名以双下划线(__)开头的方法,就会将方法名改成“ClassNamemethodName”的形式。其中ClassName表示该方法所在的类名,“methodName”表示方法名。ClassName前面要加上但单下划线()前缀。

对于上面的代码,Python编译器会将“method2”方法更名为“_Personmethod2”,所以在类的外部调用“method2”方法会抛出异常。抛出异常的原因并不是“method2”方法被私有化了,而是Python编译器把“method2”的名称改为“_Personmethod2”了。当我们了解了这些背后的原理,就可以通过调用“_Personmethod2”方法来执行“method2”方法。

1 p = Person()
2 p._Person__method2()        # 正常调用“__method2”方法

本例会创建一个MyClass类,并定义两个公共的方法(getName和setName)和一个私有的方法(outName)。然后创建了MyClass类的实例,并调用了这些方法。为了证明Python编译器在编译MyClass类时做了手脚,本例还使用了inspect模块中的getmembers函数获取MyClass类中所有的成员方法,并输出方法名。很显然,“outName”被改成了“_MyClass__outName”。

 1 class MyClass:
 2     # 公共方法
 3     def getName(self):
 4         return self.name
 5     # 公共方法
 6     def setName(self, name):
 7         self.name = name
 8         # 在类的内部可以直接调用私有方法
 9         self.__outName()
10     # 私有方法
11     def __outName(self):
12         print("Name = {}".format(self.name))
13
14 myClass = MyClass()
15 # 导入inspect模块
16 import inspect
17 # 获取MyClass类中所有的方法
18 methods = inspect.getmembers(myClass, predicate=inspect.ismethod)
19 print(methods)
20 # 输出类方法的名称
21 for method in methods:
22     print(method[0])
23 print("------------")
24 # 调用setName方法
25 myClass.setName("Bill")
26 # 调用getName方法
27 print(myClass.getName())
28 # 调用“__outName”方法,这里调用了改完名后的方法,所以可以正常执行
29 myClass._MyClass__outName()
30 # 抛出异常,因为“__outName”方法在MyClass类中并不存在
31 print(myClass.__outName())

程序运行结果如下图所示。

从getmembers函数列出的MyClass类方法的名字可以看出,“_MyClassoutName”被绑定到了“outName”方法上,我们可以将“_MyClassoutName”看做是“outName”的一个别名,一旦为某个方法起了别名,那么原来的名字在类外部就不可用了。MyClass类中的getName方法和setName方法的别名和原始方法名相同,所以在外部可以直接调用getName和setName方法。

原文地址:https://www.cnblogs.com/nokiaguy/p/10247909.html

时间: 2024-10-15 00:13:11

Python从菜鸟到高手(18):类与方法的私有化的相关文章

Python从菜鸟到高手(9):条件和条件语句

1.布尔(Boolean)值和布尔变量 ??在讲条件语句之前,首先应该了解一下布尔类型.条件语句(if)需要为其指定布尔值或布尔类型的变量,才能根据条件判断是否要指定代码块中的语句.布尔值只有两个值:True和False,可以将这两个值翻译成"真"和"假". ??现在我们已经了解了布尔值是用来做什么的,但Python语言会将哪些值看做布尔值呢?其实在Python语言中,每一种类型的值都可以被解释成布尔类型的值.例如,下面的值都会被解释成布尔值中的False. Non

Python从菜鸟到高手(10):循环

  我们现在已经知道了如何使用if语句让程序沿着不同的路径执行,不过程序最大的用处就是利用CPU和GPU强大的执行能力不断重复执行某段代码,想想Google的AlphaGo与柯洁的那场人机大战,尽管表面上是人工智能的胜利,其实人工智能只是算法,人工只能算法之所以会快速完成海量的数据分享,循环在其中的作用功不可没.   对于初次接触程序设计的读者,可能还不太理解循环到底什么东西.下面先看一下循环的伪代码. 查看银行卡余额 没有发工资,等待1分钟,继续执行1 Oh,yeah,已经发工资了,继续执行4

《Python从菜鸟到高手》已经出版,购买送视频课程

好消息,<Python从菜鸟到高手>已经出版!!! ??JetBrains官方推荐图书!JetBrains官大中华区市场部经理赵磊作序!送2400分钟同步视频课程!500个案例,400道Python练习题,电子书,10万行源代码,6个实战项目! 本书完整目录 购买地址: 当当 京东 ??在2018-10-15之前在以上地址购买本书,并5分好评以及20字以上评论.除了赠送40小时Python同步视频课程外,还赠送李宁老师任意200元以内的课程或套餐,如果课程或套餐超过200元,直接不差价即可.

Python从菜鸟到高手(2):清空Python控制台

执行python命令会进入Python控制台.在Python控制台中可以用交互的方式执行Python语句.也就是执行一行Python语句,会立刻返回执行结果. ??当Python控制台输入过多的Python语句时,有时需要将这些已经输入的语句和执行结果清空,并重新开始输入Python语句.例如,下图就是一个输入了多条Python语句,并输出相应结果的Python控制台. . ??当然,如果不想看到这些Python语句和输出结果,可以一直按"回车" 键,直到Python控制台中所有的内容

Python从菜鸟到高手(4):导入Python模块

在Python代码中导入模块需要使用import语句,语法结构如下: import module_name ??引用模块中函数的语法如下: module_name.function_name ??如果在Python程序中大量使用模块中的某些函数,那么每次在调用函数时都要加上"模块名"显得有些麻烦,所以在这种情况下,可以使用from-import-语句将模块中的函数直接暴露出来.该语句的语法结构如下: from module_name import function_name ??如果要

Python从菜鸟到高手(5):数字

1 基础知识 ??Python语言与其他编程语言一样,也支持四则运算(加.减.乘.除),以及圆括号运算符.在Python语言中,数字分为整数和浮点数.整数就是无小数部分的数,浮点数就是有小数部分的数.例如,下面的代码是标准的四则运算表达式. 2 + 4 4 * 5 + 20 5.3 / 7 (30 + 2) * 12 ??如果要计算两个数的除法,不管分子和分母是整数还是浮点数,使用除法运算符(/)的计算结果都是浮点数.例如1/2的计算结果是0.5,2/2的计算结果是1.0.要想让Python解释

Python从菜鸟到高手(7):字符串

1. 单引号字符串和转义符   字符串与数字一样,都是值,可以直接使用,在Python控制台中直接输入字符串,如"Hello World",会按原样输出该字符串,只不过用单引号括了起来. >>> "Hello World" 'Hello World'   那么用双引号和单引号括起来的字符串有什么区别呢?其实没有任何区别.只不过在输出单引号或双引号时方便而已.例如,在Python控制台输入'Let's go!',会抛出如下的错误. >>&

Python从菜鸟到高手(6):获取用户输入、函数与注释

1. 获取用户输入   要编写一个有实际价值的程序,就需要与用户交互.当然,与用户交互有很多方法,例如,GUI(图形用户接口)就是一种非常好的与用户交互的方式,不过我们先不讨论GUI的交互方式,本节会采用一种原始,但很有效的方式与用户交互,这就是命令行交互方式,也就是说,用户通过命令行方式输入数据,程序会读取这些数据,并做进一步的处理.   从命令行接收用户的输入数据,需要使用input函数.input函数接收一个字符串类型的参数,用于作为输入的提示.input函数的返回值就是用户在命令行中录入

Python从菜鸟到高手(8):print函数、赋值与代码块

1.神奇的print函数 ??print函数相信读者一定对它不陌生,因为在前面的章节,几乎每个例子都使用了print函数,这个函数的功能就是在控制台输出文本.不过print在输出文本时还可以进行一些设置,以及输出多参数字符串. ??如果为print函数传入多个参数值,那么print函数会将所有的参数值首尾相接输出. # 输出结果:a b c d e print("a","b","c","d","e"); ?

Python从菜鸟到高手(11):列表的基础操作

  列表可以使用所有适用于序列的标准操作,例如索引.分片.连接和乘法.但列表还有一些属于自己的操作,如修改列表本身的操作,这些操作包括元素赋值.元素删除.分片赋值以及下一节要将的列表方法. 1. 列表元素赋值   如果要修改列表中的某一个元素,可以像使用数组一样对列表中的特定元素赋值,也就是使用一对中括号指定元素在列表中的索引,然后使用赋值运算符(=)进行赋值.   本例修改了列表s中的前两个元素值. s = ["Bill", "Mike", "John&