天天看点

python中变量前面加单下划线和双下划线的区别&python的私有属性

python中变量前面加单下划线和双下划线的区别&python的私有属性

一、在py文件中

变量名字前面加单下划线和双下划线都是对变量的保护,一种约定,用来指定变量私有。

程序员用来指定私有变量的一种方式.不能用from module import * 导入。

在变量所属.py文件中可以正常调用。

num = 1
_num = 10
__num = 100
 
if __name__ == '__main__':
    print(num)
    print(_num)
    print(__num)
 
# 结果:
1
10
100      

二、在类里面:

1、单下划线开头的变量:

变量前的单下划线表示表面上私有 ;

但是其实这样的实例变量外部是可以访问的,但是,按照约定俗成的规定,当你看到这样的变量时,意思就是,“虽然我可以被访问,但是,请把我视为私有变量,不要随意访问”。

class Dog(object):
    _colour = '黑色'
    age = 2
 
 
dog = Dog()
print(dog._colour)  # 对象点属性的时候虽然不会自动提示,但是你硬写上去也能打印出来
 
# 结果
黑色      

2、双下划线开头的变量:

双下划线开头的变量是类里面的私有变量,只能在类的内部访问,在外部是不允许访问的和修改的

class Dog(object):
    _colour = '黑色'
    __age = 2
 
 
dog = Dog()
print(dog.__age)
 
# 结果报错
AttributeError: 'Dog' object has no attribute '__age'      

但是可以使用方法去间接的获取和修改

class Dog(object):
    _colour = '黑色'
    __age = 2
 
    def get_age(self):
        return self.__age
 
    def set_age(self, age):
        self.__age = age
 
 
dog = Dog()
print(dog.get_age())
 
dog.set_age(6)
 
print(dog.get_age())
 
# 结果
2
6      

其实不用间接获取也能获取到,python只是把私有变量的名字给修改了

class Dog(object):
    _colour = '黑色'
    __age = 2
 
    def get_age(self):
        return self.__age
 
    def set_age(self, age):
        self.__age = age
 
 
dog = Dog()
 
print(dog._Dog__age)  # 使用 _类名 + 变量名 当作属性去访问依然能访问到
 
# 结果
2      

所以,python里面没有真正意义上的私有变量

三、_xxx与__xxx与__xxx__的区别

  • xx: 公有变量
  • _x: 单前置下划线,私有化属性或方法,禁止通过from modules import *导入,但是类对象和子类可以访问
  • __xx:双前置下划线,避免与子类中的属性命名冲突,无法在外部直接访问(名字重整所以访问不到),类对象和子类不能访问
  • __xx__:双前后下划线,用户名字空间的魔法对象或属性。例如:__init__ , __ 尽量不要自定义这种形式的。
  • xx_:单后置下划线,用于避免与Python关键词的冲突

"单下划线" 开始的成员变量叫做保护变量,意思是只有类对象和子类对象自己能访问到这些变量;

"双下划线" 开始的是私有成员,意思是只有类对象自己能访问,连子类对象也不能访问到这个数据。

以双下划线开头和结尾的(__foo__)代表python里特殊方法专用的标识,如 __init__()代表类的构造函数。

四、__可以私有化属性或方法的原因

class Person(object):
    def __init__(self,name,age):
        self.__name = name
        self.age = age
 
p1 = Person("Tom",19)
#print(p1.name),#报错,因为私有属性不能直接访问,那么为什么加了__就不可访问了呢?
print(p1.age)
print(p1.__dict__) #{'_Person__name': 'Tom', 'age': 19}      

其实__方式的私有属性不可访问的原理很简单,当你在类中定义一个变量为私有变量时,python自动会将此属性名字改掉;

在类的外部使用__dict__显示一个实例所有属性,可以发现name被改成了_Person__name,所以在字典中访问name报错