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获取文件所在目录和文件名的方法
Jan 12 Python
python中matplotlib的颜色及线条控制的示例
Mar 16 Python
python模拟表单提交登录图书馆
Apr 27 Python
python pandas 对series和dataframe的重置索引reindex方法
Jun 07 Python
使用python对文件中的单词进行提取的方法示例
Dec 21 Python
python多线程http压力测试脚本
Jun 25 Python
Django 源码WSGI剖析过程详解
Aug 05 Python
Python中调用其他程序的方式详解
Aug 06 Python
python实现简单飞行棋
Feb 06 Python
Python+unittest+requests 接口自动化测试框架搭建教程
Oct 09 Python
自己搭建resnet18网络并加载torchvision自带权重的操作
May 13 Python
浅谈Python数学建模之线性规划
Jun 23 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 神盾解密工具 ”
2014/06/20 PHP
编译PHP报错configure error Cannot find libmysqlclient under usr的解决方法
2014/06/27 PHP
PHP命名空间(namespace)的动态访问及使用技巧
2014/08/18 PHP
动态的绑定事件addEventListener方法的使用
2014/01/24 Javascript
javascript中interval与setTimeOut的区别示例介绍
2014/03/14 Javascript
js严格模式总结(分享)
2016/08/22 Javascript
AngularJS 单元测试(二)详解
2016/09/21 Javascript
easyui datebox 时间限制,datebox开始时间限制结束时间,datebox截止日期比起始日期大的实现代码
2017/01/12 Javascript
用js制作淘宝放大镜效果
2020/10/28 Javascript
ES6学习之变量的解构赋值
2017/02/12 Javascript
前端开发之CSS原理详解
2017/03/11 Javascript
vue-cli webpack 引入jquery的方法
2018/01/10 jQuery
通过一次报错详细谈谈Point事件
2018/05/17 Javascript
vue 中 beforeRouteEnter 死循环的问题
2019/04/23 Javascript
微信小程序后台持续定位功能使用详解
2019/08/23 Javascript
Python查询阿里巴巴关键字排名的方法
2015/07/08 Python
Python监控主机是否存活并以邮件报警
2015/09/22 Python
Python OpenCV实现图片上输出中文
2018/01/22 Python
TensorFlow实现卷积神经网络
2018/05/24 Python
Python图像处理之图片文字识别功能(OCR)
2019/07/30 Python
Python队列RabbitMQ 使用方法实例记录
2019/08/05 Python
Python中bisect的使用方法
2019/12/31 Python
python shell命令行中import多层目录下的模块操作
2020/03/09 Python
python logging.info在终端没输出的解决
2020/05/12 Python
Matplotlib中%matplotlib inline如何使用
2020/07/28 Python
Python+OpenCV检测灯光亮点的实现方法
2020/11/02 Python
python如何用matplotlib创建三维图表
2021/01/26 Python
介绍一下Cookie和Session及他们之间的区别
2012/11/20 面试题
广播电视新闻学专业应届生求职信
2013/10/08 职场文书
给排水工程师岗位职责
2013/11/21 职场文书
综合办公室主任岗位职责
2014/04/13 职场文书
班长竞选演讲稿
2014/04/24 职场文书
敬老院标语
2014/06/27 职场文书
调研汇报材料范文
2014/08/17 职场文书
党支部创先争优活动总结
2014/08/28 职场文书
Pycharm 如何设置HTML文件自动补全代码或标签
2021/05/21 Python