Python 访问限制 private public的详细介绍


Posted in Python onOctober 16, 2018

 一、知识点

在一个模块中,我们可能会定义很多函数和变量。但有的函数和变量我们希望能给别人使用,有的函数和变量我们希望仅仅在模块内部使用,so?
我们可以通过定义该函数、变量是公开的还是私有的来达到该目的。
在Python中,是通过下划线“_”前缀来实现的。

  • public:公开的。正常的函数和变量名为此类型,可以被直接引用。比如变量abc、PI等;
  • 特殊变量:格式为__xxx__ ,以__开头、以__结尾。可以直接被引用,但是有特殊用途。比如 __author__ 、__name__就是特殊变量。一般自己定义的变量不要用这种变量名。
  • private:私有的、非公开的,格式类似于_xxx_ 和__xxx,例如__num。

不应该被直接引用,只有内部可以访问,外部不能访问。

不能随意修改对象内部的状态,这样通过访问限制的保护,代码更加健壮。

二、举例说明

在Class类内部,可以有属性和方法。而外部代码可以通过直接调用实例变量的方法来操作数据,隐藏了内部复杂逻辑。但是,外部代码还是可以自由地修改一个实例的属性。例如:

>>>b.score
99
>>>b.score = 59
>>>b.score
59

如果要让内部属性不被外部访问,可以把属性的名称前加上两个下划线“__”,变成私有变量,如下:

class Student(object):   
  def __init__(self, name, score):     
    self.__name = name     
    self.__score = score   
    
  def print_score(self):     
    print('%s: %s' % (self.__name, self.__score))

尝试在外部对属性进行访问,发现会报错,因为私有变量,不能被外部访问。

>>> bart = Student('Bart Simpson', 98) 
>>> bart.__name # 私有变量:不能被外部访问
Traceback (most recent call last):  
File "<stdin>", line 1, in <module> 
AttributeError: 'Student' object has no attribute '__name'

但是,如果外部代码要获取name和score怎么办?

给Student类增加获取属性的方法:get_name()和get_score(),如下:

class Student(object):
  ...
  def get_name(self):     
    return self.__name   
  def get_score(self):     
    return self.__score

如果外部代码修改score怎么办?可以再给Student类增加设置方法:set_score():

...
def set_score(self, score): 
  # 避免传入无效参数 
  if 0 <= score <= 100: 
    self.__score = score 
  else:       
    raise ValueError('bad score')

那作为双下划线开头的私有实例变量是不是一定不能从外部访问呢?其实也不是。

不能直接访问__name是因为Python解释器对外把__name变量改成了_Student__name,所以仍然可以通过_Student__name来访问__name变量。

>>> bart = Student('Bart Simpson', 98)
>>> bart.get_name() 
'Bart Simpson' 
>>> bart.__name = 'New Name' # 给bart新增的__name变量 
>>> bart.__name        # !与class内部的__name变量不是一个变量!
'New Name' 
>>> bart.get_name()      # get_name()内部返回self.__name (_Student__name)
'Bart Simpson'

表面上看,外部代码“成功”地设置了__name变量,但实际上这个__name变量和class内部的__name变量不是一个变量!内部的__name变量已经被Python解释器自动改成了_Student__name,而外部代码给bart新增了一个__name变量。

所以python并没有一种方法可以完全限制访问private的函数或变量,所以不是“不能被直接引用”,从编程的习惯上不应该引用private函数或变量。那他们的用处呢?

例如:

def _private_1 (name):
  return 'hello,%s ' % name
def _private_2 (name):
  return 'hi , %s ' % name
def greeting(name):
  if len(name) > 3:
    return _private_1 (name)
  else:
    return _private_2 (name)

在模块里公开greeting()函数,而把内部逻辑用private函数隐藏起来了。这样,调用greeting()函数不用关心内部的私有函数的细节。

这是一种非常有用的代码封装和抽象的方法,即:外部不需要引用的函数全部定义成private,只有外部需要引用的函数才定义为public。

三、完整代码

class Student(object):   
  def __init__(self, name, score):     
    self.__name = name     
    self.__score = score   
    
  def print_score(self):     
    print('%s: %s' % (self.__name, self.__score))
  
  def get_name(self):     
    return self.__name   
  
  def get_score(self):     
    return self.__score
    
  def set_score(self, score): 
    # 避免传入无效参数 
    if 0 <= score <= 100: 
      self.__score = score 
    else:       
      raise ValueError('bad score')  
      
  def _private_1 (name):
    return 'hello,%s ' % name
  def _private_2 (name):
    return 'hi , %s ' % name
  def greeting(name):
    if len(name) > 3:
      return _private_1 (name)
    else:
      return _private_2 (name)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python实现哈希表
Feb 07 Python
Python计算一个文件里字数的方法
Jun 15 Python
Python的Flask框架中使用Flask-SQLAlchemy管理数据库的教程
Jun 14 Python
Python实现PS滤镜的万花筒效果示例
Jan 23 Python
Python实现matplotlib显示中文的方法详解
Feb 06 Python
Python参数解析模块sys、getopt、argparse使用与对比分析
Apr 02 Python
Python爬虫——爬取豆瓣电影Top250代码实例
Apr 17 Python
python matplotlib拟合直线的实现
Nov 19 Python
python实现扫雷游戏
Mar 03 Python
PyCharm Anaconda配置PyQt5开发环境及创建项目的教程详解
Mar 24 Python
通过cmd进入python的步骤
Jun 16 Python
Python自动发送和收取邮件的方法
Aug 12 Python
对django中render()与render_to_response()的区别详解
Oct 16 #Python
浅谈django的render函数的参数问题
Oct 16 #Python
django之跨表查询及添加记录的示例代码
Oct 16 #Python
python看某个模块的版本方法
Oct 16 #Python
对python中的 os.mkdir和os.mkdirs详解
Oct 16 #Python
详解Django的model查询操作与查询性能优化
Oct 16 #Python
python查看模块,对象的函数方法
Oct 16 #Python
You might like
php的一些小问题
2010/07/03 PHP
浅析php学习的路线图
2013/07/10 PHP
php判断用户是否关注微信公众号
2016/07/22 PHP
初学JavaScript第二章
2008/09/30 Javascript
javaScript 数值型和字符串型之间的转换
2009/07/25 Javascript
JavaScript生成福利彩票双色球号码
2015/05/15 Javascript
JavaScript、tab切换完整版(自动切换、鼠标移入停止、移开运行)
2016/01/05 Javascript
JavaScript中通过提示框跳转页面的方法
2016/02/14 Javascript
JQuery+EasyUI轻松实现步骤条效果
2016/02/22 Javascript
js判断图片加载完成后获取图片实际宽高的方法
2016/02/25 Javascript
js和jquery中获取非行间样式
2017/05/05 jQuery
Angular2入门教程之模块和组件详解
2017/05/28 Javascript
9种改善AngularJS性能的方法
2017/11/28 Javascript
基于Vue-Cli 打包自动生成/抽离相关配置文件的实现方法
2018/12/09 Javascript
layer.alert自定义关闭回调事件的方法
2019/09/27 Javascript
详解javascript中var与ES6规范中let、const区别与用法
2020/01/11 Javascript
原生JavaScript之es6中Class的用法分析
2020/02/23 Javascript
Python 文件重命名工具代码
2009/07/26 Python
python使用wmi模块获取windows下硬盘信息的方法
2015/05/15 Python
python用列表生成式写嵌套循环的方法
2018/11/08 Python
在python中使用with打开多个文件的方法
2019/01/07 Python
Python中的几种矩阵乘法(小结)
2019/07/10 Python
python+openCV调用摄像头拍摄和处理图片的实现
2019/08/06 Python
python django生成迁移文件的实例
2019/08/31 Python
Python之Numpy的超实用基础详细教程
2019/10/23 Python
python3中numpy函数tile的用法详解
2019/12/04 Python
用openCV和Python 实现图片对比,并标识出不同点的方式
2019/12/19 Python
PHP两种查询函数array/row的区别
2013/06/03 面试题
省级优秀毕业生主要事迹
2014/05/29 职场文书
经理任命书模板
2014/06/06 职场文书
党员教师群众路线个人整改措施
2014/10/28 职场文书
夫妻分居协议书范文
2014/11/26 职场文书
神农溪导游词
2015/02/11 职场文书
2016年共产党员公开承诺书
2016/03/24 职场文书
小学2016年第十八届推普周活动总结
2016/04/05 职场文书
Redis sentinel哨兵集群的实现步骤
2022/07/15 Redis