在python中单下划线代表私有,但也仅仅是名义上的私有,只是一种规范,告诉人们不要在外部使用它。但实际上python没有真正意义上的私有,我们一样可以在外部去调用私有方法或属性。
class BaseForm(StrAndUnicode): def _get_errors(self): ‘‘‘ Returns an ErrorDict for the data provided for the form :return: ‘‘‘ if self._errors is None: self.full_clean() return self._errors errors = property(_get_errors)
该代码片段来自Django源码(django/forms/forms.py)。这段代码的设计就是errors属性是对外API的一部分,如果你想获取错误详情,应该访问errors属性,而不是(也不应该)访问_get_errors方法。实际上我们仍然可以使用对象在外部调用它。
baseform = BaseForm()
baseform._get_errors()
双下划线使用来避免父类方法被子类方法覆盖的。双下划线方法的本质是在方法前加了_类名,我们可以使用对象._类名__方法名(),来在外部调用它。
class A(object): def __method(self): print("I‘m a method in class A") def method_x(self): print("I‘m another method in class A\n") def method(self): self.__method() self.method_x() class B(A): def __method(self): print("I‘m a method in class B") def method_x(self): print("I‘m another method in class B\n") if __name__ == ‘__main__‘: print("situation 1:") a = A() a.method() b = B() b.method() print("situation 2:") #b.__method() 报错 ‘B‘ object has no attribute ‘__method‘ a._A__method()
上面程序的执行结果为:
situation 1: I‘m a method in class A I‘m another method in class A I‘m a method in class A I‘m another method in class B situation 2: I‘m a method in class A
前后双下滑线方法是python自己定义出来,供自己调用的。这些方法会在特定的条件下被触发执行。下面是常用的一些前后双下划线方法。
__str__ 当将对象转换成字符串时会执行 __init__ 初始化方法,为对象变量赋值 __new__ 构造方法,创建一个对象 __call__ 在对象后面加括号会执行该方法 __getattr__ 当使用对象.属性时,若属性不存在会调用该方法 __setattr__ 当使用对象.属性 = 值,会调用该方法 __iter__ 类内部定义了该方法,对象就变成了可迭代对象 __add__ 当两个对象使用+号会调用该方法 __enter__和__exit__ 上下文管理 class Foo(object): def __enter__(self): print(‘开始‘) return Foo() def __exit__(self, exc_type, exc_val, exc_tb): print(‘结束‘) with Foo() as obj: #with后面的Foo()会执行__enter__()方法,并将返回值赋给obj print(123) #缩进代码执行完成后会执行__exit__()方法 #开始 #123 #结束
我们可以在自己的类中重写者些方法,以便实现特定的功能。
以上大部分内容参考自https://www.jb51.net/article/129534.htm
原文:https://www.cnblogs.com/wang-kai-xuan/p/11407832.html