Python 私有属性 & 私有方法

liftword23小时前技术文章3

1. 场景定义

私有属性

是指在 Python 的面向对象开发过程中,对象的某些属性只想在对象的内部被使用,但不想在外部被访问到这些属性。

即:私有属性是对象不愿意公开的属性。


私有方法

是指在 Python 的面向对象开发过程中,对象的某些方法或者称为函数只想在对象的内部被使用,但不想在外部被访问到这些方法或函数。

即:私有方法是对象不愿意公开的方法或函数。


2. 语法定义

在 Python 中定义私有属性和私有方法的语法如下:

class Staff:
    def __init__(self, s_name, s_salary):
      self.s_name = s_name
      self.__salary = s_salary
      
    def __secret(self):
      print("%s 的工资是 %d" % (self.s_name, self.__salary))

(1). __salary是以两个下划线开头来定义的私有属性

(2). __secret(self)是以两个下划线开头来定义的私有方法


3. 调用分析

(1). 在__init__的对象初始化方法中,以两个下划线开头定义的__salary属性就是私有属性。

现在在对象的外部来调用一下__salary属性,看是否能正常访问该私有属性。

从上图运行结果可以看出,第11行,即在对象外部访问对象的私有属性 __salary 时,提示 AttributeError 错误,Staff 对象 zhangsan 没有属性 __salary。


为了证明 Staff 类对象确实是有__salary 这个实例属性的,只是因为在对象外部不能访问私有属性。


我把 self.__salary 修改为:self.salary,__secret(self)方法对self.__salary属性的引用,做相应的修改,看如下图所示的运行结果。

可以从运行结果看出,这种非私有属性在外部的调用是正常的,没有提示 AttributeError 错误。


(2). 在 __secret(self) 实例方法中,以两个下划线开头定义的__secret(self)方法就是私有方法。

和上面测试流程一样,先在对象的外部来调用私有方法__secret(self),看是否能正常调用该私有方法。

从上图运行结果可以看出,第11行,即在对象外部访问对象的私有方法 __secret(self) 时,提示 AttributeError 错误,Staff 对象 zhangsan 没有 __secret 方法。


为了证明 Staff 类对象是有__secret(self)这个实例方法的,只是因为在对象外部不能访问私有方法。


我把 __secret(self) 方法修改为:secret(self),其他代码不变,看如下图所示的运行结果。

可以从运行结果看出,这种非私有方法在外部的调用是正常的,没有提示 AttributeError 错误。


(3). 从下图可以看出,在对象内部私有方法与私有属性是可以被调用的。

如图中的 work 方法调用了私有方法__secret(self),而私有方法__secret(self)调用了私有属性__salary。

在对象外部使用 Staff 类对象 zhangsan 来调用 work 方法,可以间接访问到对象的私有属性和私有方法。

从控制台输出结果来看 work 方法能正常访问到对象内部定义的私有属性和私有方法。


4. Python伪私有属性和私有方法

在 Python 中,并没有真正意义上的私有,因为 Python 内部在给属性、方法命名时,对名称做了一些特殊处理,使得外界无法访问到对应的属性和方法。

以私有属性和私有方法为例,Python内部处理方式为:

(1). 属性: __salary,经过处理后的属性名为:_Staff__salary(_类名__属性名)

(2). 方法: __secret,经过处理后的方法名为:_Staff__secret(_类名__方法名)

知道了 Python 内部对于私有属性和私有方法的处理,现在使用这种处理后的命名方式来在对象外部访问私有属性和私有方法,看是否能访问正常。

class Staff:
    def __init__(self, s_name, s_salary):
        self.s_name = s_name
        self.__salary = s_salary

    def __secret(self):
        return "%s的工资是 %d" % (self.s_name, self.__salary)


zhangsan = Staff("张三", 10000)
print(zhangsan._Staff__salary)
print(zhangsan._Staff__secret())

运行结果如下图所示

控制台没有抛任何的异常,之前的提示 AttributeError 错误也没有了。

这个例子证明了 Python 是没有真正意义上的私有的,当知道了其内部处理方式后,依然可以使用_类名__属性名(方法名)的方法来在对象外部访问到对象内部定义的私有属性和私有方法。


但这种方式在日常工作中是不推荐使用的,既然在对象内部定义属性和方法时,就声明了其为私有的,调用方就需要遵守其规则。

这里只是想通过这个小例子来说明 Python 并无真正意义上的私有。

相关文章

Python | 一文搞懂self

关于self在Python中,self是一个关键字,它表示类实例本身。通常,self是在类方法中的第一个参数传递的。通过使用self,我们可以访问类的属性和方法。当一个类定义后,我们可以使用它来创建对...

python self是什么鬼,其本质就是特殊的形参

小伙伴们能真正理解 self 是什么东西吗?现在我从实质上为小伙伴们讲解一下 self。self 的实质是形参,现通过普通形参,来讲解特殊形参 self看基本概念的理解普通形参:通过 函数(参数) 方...

为什么python的self参数必须显式声明?隐式this才是万恶之源

杂谈python中存在 self 这个参数,记得有个朋友是写 java 的,当他看到python中的代码立刻明白了它的用途。他问我:“为什么python中的 self不能像this一样,隐式声明多好,...

Python这个代码里面x=self是什么意思啊

大家好,我是Python进阶者。一、前言前几天在Python白银交流群【无敌劈叉小狗】问了一个Python基础的问题,问题如下:这个代码里面x=self是什么意思啊?self到底是个什么存在呢?感觉把...

Python super()函数:调用父类的构造方法

前面不止一次讲过,Python 中子类会继承父类所有的类属性和类方法。严格来说,类的构造方法其实就是实例方法,因此毫无疑问,父类的构造方法,子类同样会继承。但我们知道,Python 是一门支持多继承的...

一文带你解读JavaScript的引用类型和函数对象

前言相信做网站对JavaScript再熟悉不过了,它是一门脚本语言,不同于Python的是,它是一门浏览器脚本语言,而Python则是服务器脚本语言,我们不光要会Python,还要会JavaScrip...