谈谈Python:为什么类中的私有属性可以在外部赋值并访问


Posted in Python onMarch 05, 2020

Python:为什么类中的私有属性可以在外部赋值并访问?

问题引入

在慕课网上学习Python**类中的私有属性**的时候,看到了一个同学的提问:

将count改为__count,为什么实例变量在外部仍然可以修改__count?这里print p1.__count可以打印出100

class Person(object):
 __count = 0
 def __init__(self, name):
  Person.__count = Person.__count + 1
  self.name = name
  print Person.__count
 p1 = Person('Bob')
 p1.__count=100
 print p1.__count
 p2 = Person('Alice')

print Person.__count

问题解决:

单刀直入版:

这是因为给p1.__count赋值的操作,其实是在p1中定义了一个名为__count的变量(因为Python中的都是动态变量),而没有改变类中真正的属性。

太长但还是要看看版:

知识点清单:

1、类的“伪私有属性”
2、在类的外部动态地创建类属性

问题解决过程:

1、“伪私有属性”的概念:

python的类中通过加双下划线来设置的“私有属性”其实是“伪私有属性”,原理是python编译器将加了双下划线的“属性名”自动转换成“类名属性名”。所以我们在外部用“属性名”访问私有属性的时候,会触发AttributeError,从而实现“私有属性”的特性。但通过“类名属性名”也可以访问这些属性。

参考:http://www.pythonclub.org/python-class/private

2、编写测试代码:

以下是在该同学的代码的基础上修改的测试代码:

class Person(object):
 #设置类属性
 __count_of_class = 'original count_of_class'
 def __init__(self, name):
  self.name = name
  print('in class Person : count_of_class = ', Person.__count_of_class,'\n')

#初始化实例p1
p1 = Person('Bob')
#在实例p1上修改属性值
p1.__count_of_class='I\'m not the original count_of_class!'
print('p1\'s _Person__count_of_class = ',p1._Person__count_of_class)
print('p1\'s __count_of_class = ',p1.__count_of_class,'\n')

#在类Person上修改属性值
Person.__count_of_class = 'I\'m not the original count_of_class!'
#将这句注释取消掉,会发现真正的私有属性的值也改变了
#Person._Person__count_of_class = 'I\'m not the original count_of_class!'
print('Person\'s _Person__count_of_class = ',Person._Person__count_of_class)
print('Person\'s __count_of_class = ',Person.__count_of_class)

分别在实例p1上和类Person上进行操作,并且分别打印出“__属性名”,以及“_类名__属性名”。

输出结果如下:

in class Person : count_of_class = original count_of_class

p1's _Person__count_of_class = original count_of_class
p1's __count_of_class = I'm not the original count_of_class!

Person's _Person__count_of_class = original count_of_class
Person's __count_of_class = I'm not the original count_of_class!

**由此可见,虽然用p1.__count_of_class给它赋值了,但其实在类中真正的属性_Person__count_of_class的原始值是没有改变的。

但是如果将p1._Person__count_of_class赋值,那么类属性定义的原始值就真正地被覆盖了**

"""
取消掉
##Person._Person__count_of_class = 'I\'m not the original count_of_class!'
的注释,输出结果:
"""

in class Person : count_of_class = original count_of_class 
p1's _Person__count_of_class = original count_of_class 
p1's __count_of_class = I'm not the original count_of_class! 

#注意这一句:
Person's _Person__count_of_class = I'm not the original count_of_class! 
Person's __count_of_class = I'm not the original count_of_class!

由此,我们知道了:_count_of_class和_Person_count_of_class不是同一个东西。

最后的问题

但是呢,如果不先给p1.__count_of_class赋值,直接打印它又会触发AttributeError,这是为什么?

这是因为给p1.__count_of_class赋值的操作,其实是在p1中定义了一个名为__count_of_class的变量(因为Python中的都是动态变量)。

以下实例说明可以通过外部赋值来为类创造属性:

class Person(object):
 pass

p1=Person()
#给p1创建属性new_of_instance
p1.new_of_instance = 'I\'m new in p1!'
print(p1.new_of_instance)

#给Person类创建属性new_of_class
Person.new_of_class = 'I\'m new in Person!'

#在类中新加的属性,可以通过实例来访问
print(p1.new_of_class)


>>>输出:
I'm new in p1!
I'm new in Person!

问题解决。

以上这篇谈谈Python:为什么类中的私有属性可以在外部赋值并访问就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python使用urllib模块和pyquery实现阿里巴巴排名查询
Jan 16 Python
python统计文本文件内单词数量的方法
May 30 Python
Python中的条件判断语句基础学习教程
Feb 07 Python
Python+request+unittest实现接口测试框架集成实例
Mar 16 Python
可能是最全面的 Python 字符串拼接总结【收藏】
Jul 09 Python
Python常见排序操作示例【字典、列表、指定元素等】
Aug 15 Python
pytorch对可变长度序列的处理方法详解
Dec 08 Python
使用Python横向合并excel文件的实例
Dec 11 Python
通过python爬虫赚钱的方法
Jan 29 Python
Python实现的合并两个有序数组算法示例
Mar 04 Python
深度学习入门之Pytorch 数据增强的实现
Feb 26 Python
Python爬虫之Selenium库的使用方法
Jan 03 Python
python如何将两张图片生成为全景图片
Mar 05 #Python
Python 定义只读属性的实现方式
Mar 05 #Python
Pycharm中import torch报错的快速解决方法
Mar 05 #Python
Python中私有属性的定义方式
Mar 05 #Python
Python实现AI自动抠图实例解析
Mar 05 #Python
python GUI库图形界面开发之PyQt5 MDI(多文档窗口)QMidArea详细使用方法与实例
Mar 05 #Python
Python matplotlib修改默认字体的操作
Mar 05 #Python
You might like
深入PHP获取随机数字和字母的方法详解
2013/06/06 PHP
PHP中mb_convert_encoding与iconv函数的深入解析
2013/06/21 PHP
php readfile()修改文件上传大小设置
2017/08/11 PHP
基于Jquery的淡入淡出的特效基础练习
2010/12/13 Javascript
escape函数解决js中ajax传递中文出现乱码问题
2014/10/30 Javascript
JavaScript中的toDateString()方法使用详解
2015/06/12 Javascript
JavaScript的Backbone.js框架的一些使用建议整理
2016/02/14 Javascript
jquery实现简单的banner轮播效果【实例】
2016/03/30 Javascript
chorme 浏览器记住密码后input黄色背景处理方法(两种)
2017/11/22 Javascript
关于Vue的路由权限管理的示例代码
2018/03/06 Javascript
jQuery实现文件编码成base64并通过AJAX上传的方法
2018/04/12 jQuery
微信小程序实现下拉框功能
2019/07/16 Javascript
详解ES6 Promise的生命周期和创建
2019/08/18 Javascript
解决layui中onchange失效以及form动态渲染失效的问题
2019/09/27 Javascript
[02:19]DOTA2上海特级锦标赛 观赛指南 Spectator Guide
2016/02/04 DOTA
Django查找网站项目根目录和对正则表达式的支持
2015/07/15 Python
Python 多线程的实例详解
2017/09/07 Python
Python+matplotlib实现计算两个信号的交叉谱密度实例
2018/01/08 Python
Python cookbook(数据结构与算法)从字典中提取子集的方法示例
2018/03/22 Python
Python Grid使用和布局详解
2018/06/30 Python
python实现将字符串中的数字提取出来然后求和
2020/04/02 Python
TensorFlow2.1.0最新版本安装详细教程
2020/04/08 Python
python中remove函数的踩坑记录
2021/01/04 Python
全网最细 Python 格式化输出用法讲解(推荐)
2021/01/18 Python
CSS3过渡transition效果实例介绍
2016/05/03 HTML / CSS
荷兰浴室和卫浴网上商店:Badkamerxxl.nl
2020/10/06 全球购物
创新比赛获奖感言
2014/02/13 职场文书
暑期社会实践感言
2014/02/25 职场文书
坚定理想信念心得体会
2014/03/11 职场文书
违反交通法规检讨书
2014/09/10 职场文书
交通事故赔偿协议书怎么写
2014/10/04 职场文书
观后感开头
2015/06/19 职场文书
写给同事的离职感言
2015/08/04 职场文书
python 如何在list中找Topk的数值和索引
2021/05/20 Python
Vue中插槽slot的使用方法与应用场景详析
2021/06/08 Vue.js
python模块与C和C++动态库相互调用实现过程示例
2021/11/02 Python