四周二次课(11月7日)

一、 类的重写

1.1 重写一般方法

class A:    def hello(self):        print('Hello,i am A.')class B(A):    passa = A()b = B()a.hello()b.hello()

结果:

Hello,i am A.Hello,i am A.

解释:

B类没有定义自己的hello方法,故当hello被调用时,原始信息就被打印出来了。

B类也可以重写这个hello方法

class A:    def hello(self):        print('Hello,i am A.')class B(A):    def hello(self):          print('Hello,i am B.')a = A()b = B()a.hello()b.hello()

结果:

Hello,i am A.Hello,i am B.

1.2 重写特殊的构造方法

1、调用未绑定的基类构造方法

class Bird:    def __init__(self):        self.hungry = True    def eat(self):        if self.hungry:            print('Aaaah...')            self.hungry = False        else:            print('No,thanks!')class SongBird(Bird):    def __init__(self):            Bird.__init__(self)            self.sound = 'Squawk!'    def sing(self):        print(self.sound)b = SongBird()b.sing()b.eat()b.eat()

结果:

Squawk!Aaaah...No,thanks!

2、使用super函数

class Bird():    def __init__(self):        self.hungry = True    def eat(self):        if self.hungry:            print('Aaaah...')            self.hungry = False        else:            print('No,thanks!')class SongBird(Bird):    def __init__(self):        super(SongBird,self).__init__()        self.sound = 'Squawk!'    def sing(self):        print(self.sound)b = SongBird()b.sing()b.eat()b.eat()

结果:

Squawk!Aaaah...No,thanks!

解释:

Python 2中定义类需要写上object,否则报错。

二、类的私有变量

在Python中可以通过在属性变量名前加上双下划线定义属性为私有属性

特殊变量命名

1、 _xx 以单下划线开头的表示的是protected类型的变量。即保护类型只能允许其本身与子类进行访问。若内部变量标示,如: 当使用“from M import”时,不会将以一个下划线开头的对象引入 。

2、 __xx 双下划线的表示的是私有类型的变量。只能允许这个类本身进行访问了,连子类也不可以用于命名一个类属性(类变量),调用时名字被改变(在类FooBar内部,__boo变成_FooBar__boo,如self._FooBar__boo)

3、 __xx__定义的是特殊方法。用户控制的命名空间内的变量或是属性,如init , __import__或是file 。只有当文档有说明时使用,不要自己定义这类变量。 (就是说这些是python内部定义的变量名)

在这里强调说一下私有变量,python默认的成员函数和成员变量都是公开的,没有像其他类似语言的public,private等关键字修饰.但是可以在变量前面加上两个下划线"_",这样的话函数或变量就变成私有的.这是python的私有变量轧压(这个翻译好拗口),英文是(private name mangling.) **情况就是当变量被标记为私有后,在变量的前端插入类名,再类名前添加一个下划线"_",即形成了_ClassName__变量名.**

Python内置类属性

__dict__ : 类的属性(包含一个字典,由类的数据属性组成)

__doc__ :类的文档字符串

__module__: 类定义所在的模块(类的全名是'__main__.className',如果类位于一个导入模块mymod中,那么className.__module__ 等于 mymod)

__bases__ : 类的所有父类构成元素(包含了一个由所有父类组成的元组)

例子1:

class A(object):     def __init__(self):         self.__data = []  # 翻译成 self._A__data=[]     def add(self, item):         self.__data.append(item)  # 翻译成 self._A__data.append(item)     def printData(self):         print self.__data  # 翻译成 self._A__data a = A() a.add('hello') a.add('python') a.printData() # print a.__data  #外界不能访问私有变量 AttributeError: 'A' object has no attribute '__data' print a._A__data  # 通过这种方式,在外面也能够访问“私有”变量;这一点在调试中是比较有用的!

结果:

['hello', 'python'] ['hello', 'python']

例子2:

class A():    def __init__(self):        self.__name = 'python'  # 私有变量,翻译成 self._A__name='python'    def __say(self):  # 私有方法,翻译成 def _A__say(self)        print self.__name  # 翻译成 self._A__namea = A()# print a.__name #访问私有属性,报错!AttributeError: A instance has no attribute '__name'print a.__dict__  # 查询出实例a的属性的集合print a._A__name  # 这样,就可以访问私有变量了# a.__say()#调用私有方法,报错。AttributeError: A instance has no attribute '__say'print dir(a)  # 获取实例的所有属性和方法a._A__say()  # 这样,就可以调用私有方法了

结果:

{'_A__name': 'python'}python['_A__name', '_A__say', '__doc__', '__init__', '__module__']python