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 03 Python
pycharm中连接mysql数据库的步骤详解
May 02 Python
Python实现判断一个字符串是否包含子串的方法总结
Nov 21 Python
Django admin美化插件suit使用示例
Dec 12 Python
Tensorflow 训练自己的数据集将数据直接导入到内存
Jun 19 Python
Python DataFrame.groupby()聚合函数,分组级运算
Sep 18 Python
如何爬取通过ajax加载数据的网站
Aug 15 Python
python读取dicom图像示例(SimpleITK和dicom包实现)
Jan 16 Python
Python通过kerberos安全认证操作kafka方式
Jun 06 Python
Python函数参数定义及传递方式解析
Jun 10 Python
keras的siamese(孪生网络)实现案例
Jun 12 Python
pandas将list数据拆分成行或列的实现
Dec 13 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
MySQL授权问题总结
2007/05/06 PHP
wamp下修改mysql访问密码的解决方法
2013/05/07 PHP
基于php的CMS中展示文章类实例分析
2015/06/18 PHP
PHP生成条形码大揭秘
2015/09/24 PHP
Yii2.0中的COOKIE和SESSION用法
2016/08/12 PHP
PHPCMS忘记后台密码的解决办法
2016/10/30 PHP
类似框架的js代码
2006/11/09 Javascript
几款极品的javascript压缩混淆工具
2007/05/16 Javascript
定义select的边框颜色
2008/04/28 Javascript
jQuery EasyUI API 中文文档 DateTimeBox日期时间框
2011/10/16 Javascript
浅谈Javascript事件模拟
2012/06/27 Javascript
JS前端框架关于重构的失败经验分享
2013/03/17 Javascript
jquery获取当前日期的方法
2015/01/14 Javascript
原生js实现类似弹窗抖动效果
2015/04/02 Javascript
jquery实现两边飘浮可关闭的对联广告
2015/11/27 Javascript
Angular学习笔记之angular的$filter服务浅析
2016/11/12 Javascript
Bootstrap源码学习笔记之bootstrap进度条
2016/12/24 Javascript
深入理解Vue-cli搭建项目后的目录结构探秘
2017/07/13 Javascript
Vue.js与 ASP.NET Core 服务端渲染功能整合
2017/11/16 Javascript
在Vue项目中引入JQuery-ui插件的讲解
2019/01/27 jQuery
详解vue-cli3开发Chrome插件实践
2019/05/29 Javascript
JS字符串补全方法padStart()和padEnd()
2020/05/27 Javascript
Python中函数的多种格式和使用实例及小技巧
2015/04/13 Python
python的unittest测试类代码实例
2017/12/07 Python
Python回文字符串及回文数字判定功能示例
2018/03/20 Python
python使用mitmproxy抓取浏览器请求的方法
2019/07/02 Python
python递归调用中的坑:打印有值, 返回却None
2020/03/16 Python
详解CSS3中nth-child与nth-of-type的区别
2017/01/05 HTML / CSS
伦敦一家西班牙童装精品店:La Coqueta
2018/02/02 全球购物
初中毕业典礼演讲稿
2014/09/09 职场文书
教师党员整改措施
2014/10/24 职场文书
社区法制宣传月活动总结
2015/05/07 职场文书
2015年财务经理工作总结
2015/05/13 职场文书
机关单位2016年创先争优活动总结
2016/04/05 职场文书
《围炉夜话》110句人生箴言,精辟有内涵,引人深思
2019/10/23 职场文书
python游戏开发之pygame实现接球小游戏
2022/04/22 Python