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 16 Python
python实现判断数组是否包含指定元素的方法
Jul 15 Python
Django web框架使用url path name详解
Apr 29 Python
java中的控制结构(if,循环)详解
Jun 26 Python
Django分页功能的实现代码详解
Jul 29 Python
python实现超市商品销售管理系统
Oct 25 Python
python如何使用socketserver模块实现并发聊天
Dec 14 Python
python多线程实现代码(模拟银行服务操作流程)
Jan 13 Python
Python with标签使用方法解析
Jan 17 Python
Jupyter安装链接aconda实现过程图解
Nov 02 Python
python如何获得list或numpy数组中最大元素对应的索引
Nov 16 Python
用Python写一个简易版弹球游戏
Apr 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
漫威DC御用漫画家去世 他的表情包曾走红网络
2020/04/09 欧美动漫
用PHP实现验证码功能
2006/10/09 PHP
一个PHP数组应该有多大的分析
2009/07/30 PHP
PHP 伪静态隐藏传递参数名的四种方法
2010/02/22 PHP
php防止sql注入示例分析和几种常见攻击正则表达式
2014/01/12 PHP
PHP 布尔值的自增与自减的实现方法
2018/05/03 PHP
laravel在中间件内生成参数并且传递到控制器中的2种姿势
2019/10/15 PHP
IE和FireFox(FF)中js和css的不同
2009/04/13 Javascript
根据出生日期自动取得星座的js代码
2010/07/20 Javascript
如何编写高质量JS代码
2014/12/28 Javascript
快速学习jQuery插件 Cookie插件使用方法
2015/12/01 Javascript
JavaScript编写带旋转+线条干扰的验证码脚本实例
2016/05/30 Javascript
JavaScript实现翻页功能(附效果图)
2017/02/16 Javascript
Vue从TodoList中学父子组件通信
2019/02/05 Javascript
vue路由拦截器和请求拦截器知识点总结
2019/11/08 Javascript
VUE中使用HTTP库Axios方法详解
2020/02/05 Javascript
Javascript ParentNode和ChildNode接口原理解析
2020/03/16 Javascript
js实现省级联动(数据结构优化)
2020/07/17 Javascript
前端性能优化建议
2020/09/17 Javascript
[06:24]DOTA2 2015国际邀请赛中国区预选赛第二日TOP10
2015/05/27 DOTA
python创建线程示例
2014/05/06 Python
python代码制作configure文件示例
2014/07/28 Python
详解Python中的装饰器、闭包和functools的教程
2015/04/02 Python
Tensorflow 利用tf.contrib.learn建立输入函数的方法
2018/02/08 Python
基于Python Numpy的数组array和矩阵matrix详解
2018/04/04 Python
对Python闭包与延迟绑定的方法详解
2019/01/07 Python
超简单的Python HTTP服务
2019/07/22 Python
Python queue队列原理与应用案例分析
2019/09/27 Python
英国现代绅士品牌:Hackett
2017/12/17 全球购物
Zadig&Voltaire官网:法国时装品牌
2018/01/05 全球购物
GC是什么?为什么要有GC?
2013/12/08 面试题
酒吧副总经理岗位职责
2013/12/10 职场文书
社会调查研究计划书
2014/05/01 职场文书
高一课前三分钟演讲稿
2014/09/13 职场文书
乡镇党的群众路线教育实践活动总结报告
2014/10/30 职场文书
银行求职自荐信范文
2015/03/04 职场文书