Python用下划线
Python用下划线为变量前缀和后缀制定特殊变量
_xxx 不能用 'from module import *' 导入
__xxx__ 系统定义名字
__xxx 类中的私有变量名
核心风格:避免用下划线作为变量名的开始
因为下划线对解释器有特殊的意义,而且内建的标识符所使用的符号,我们建议程序员避免用下划线作为变量名的开始。
一般来讲,变量名_xxx被看做是'私有的',在模块和类外不可以使用。当变量是私有的时候,用_xxx来表示变量是很好的习惯
因为变量名__xxx__对python来说有特殊含义,对于普通的变量应当避免这种命名风格。
'单下划线':开始的成员变量叫做保护变量,意思是只有类对象和子类对象自己能访问到这些变量;
'双下划线':开始的是私有成员,意思是只有类对象自己能访问,连子类对象也不能访问到这个数据。
实例:
class Foo(object): def __init__(self, name, age): self._name = name # 单下划线开始的成员变量叫做保护变量,只有类对象和子类对象自己能访问到这些变量 self.__age = age # 双下划线是私有成员,只有类的内部能调用,类对象能够访问,但是应该避免直接去访问 f = Foo('hkey', 20) print(f.__dict__) # 打印对象所有的属性和方法 print(f._name) # 执行结果: # {'_Foo__age': 20, '_name': 'hkey'} # hkey
以单下划线开头(_name)的代表不能直接访问的类属性,需通过类提供的接口进行访问,不能用'from xxx import *'导入
以双下划线开头的(__age)代表类的私有成员;
以双下划线开头和结尾的(__foo__)代表python里特殊方法专用的标识,如__init__()代表类的构造函数。
类中特殊方法的使用
__new__:类中的构造方法
__init__: 类中的初始化方法
class A(object): def __init__(self): print('init') def __new__(cls, *args, **kwargs): print('new %s' %cls) return object.__new__(cls, *args, **kwargs) A() # 执行结果: # new <class '__main__.A'> # init
知识点:
1. 继承自object的新式类才有 __new__
2. __new__至少要有一个参数cls,代表要实例化的类,此参数在实例化时由python解释器自动提供;
3. __new__必须要有返回值,返回实例化出来的实例,这点在自己实现__new__时要特别注意;
4. __init__有一个参数self,就是这个__new__返回的实例,__init__在__new__的基础上可以完成一些其他初始化的动作,
__init__不需要返回值,__new__方法一定要有返回值
5. 若__new__没有正确返回当前cls的实例,那__init__是不会被调用的,即使是父类的实例也不行;
__len__:如果类表现的像一个list,要获取有多少个元素,就得用 len()函数,要让len()函数工作正常,类必须提供一个
特殊的方法 __len__(),它返回元素的个数;
实例:
class Student(object): def __init__(self, *args): self.names = args def __len__(self): return len(self.names) s = Student('xiaofei', 'hkey', 'xiaoA') print(len(s)) # 执行结果: # 3
__str__ 和 __repr__
使用__str__和__repr__ 打印类的对象,返回的不再是一个内存地址,而是定义后字符串内容,__str__和__repr__返回的必须是字符串类型:
class Student(object): def __init__(self, name): self.name = name def __str__(self): return 'Student object (name: %s)' % self.name s = Student('hkey') print(s) # 用内置方法 str 和 repr 调用 print(str(s)) print(repr(s)) # 执行结果: # Student object (name: hkey) # Student object (name: hkey) # print(repr(s))执行结果: # <__main__.Student object at 0x000001D0EB04BC18>
在上面的实例中没有定义__repr__特殊方法,当使用内置方法 repr 调用的时候显示的是类的内存地址信息,这两者的区别:
__str__:返回用户看到的字符串
__repr__:返回程序开发者看到的字符串,也就是说__repr__()为调试使用的。
这里有个偷懒的写法:
class Student(object): def __init__(self, name): self.name = name def __str__(self): return 'Student object (name: %s)' % self.name __repr__ = __str__ s = Student('hkey') print(s) # 用内置方法 str 和 repr 调用 print(str(s)) print(repr(s)) # 执行结果: # Student object (name: hkey) # Student object (name: hkey) # print(repr(s))执行结果: # Student object (name: hkey)
在类中,定义这两个特殊方法的时候要要注意:
当使用内置方法 str 对类对象进行调用的时候,首先会去找类中的 __str__方法,如果不存在就会找 __repr__方法,
所以如果实现这两个特殊方法返回的结果一致时,简单方式就是写一个__repr__的特殊方法就行;
__call__:可以直接对实例进行调用
实例:
class Student(object): def __init__(self, name): self.name = name def __call__(self, *args, **kwargs): print('My name is %s.' % self.name) s = Student('hkey') s() # 执行结果: # My name is hkey.