类和实例
class Student(object):
def __init__(self,name,age):
self.name=name#实例变量绑定属性
self.age=age
#数据封装
def print_name(self):
print('name:%s'%self.name)
def get_grade(self):
if self.age>20:
return 'A'
elif self.age>15:
return 'B'
else:
return 'C'
jack=Student('liqian',34)
sam=Student('sam',13)
print(sam.print_name())
print(jack.name,jack.get_grade())
print(sam.age,sam.get_grade())
访问限制
如果外部要访问:def get_xxx(self):
继承:
1、子类获得父类所有方法
2、子类和父类有相同方法时,字类的方法覆盖了父类的方法
多态
在调用父类和字类相同方法时,调用的总是字类的方法
不一定需要传入Animal类型。我们只需要保证传入的对象有一个run()方法就可以了
获取对象信息
type()判断对象类型
>>> type('abc')==type(123)False
对于继承关系使用isinstance()
>>> a = Animal()
>>> d = Dog()
>>> h = Husky()
>>> isinstance(h, Dog)
True
能用type()判断的基本类型也可以用isinstance()判断
>>> isinstance('a', str)
获得一个对象的所有属性和方法,使用dir()函数,它返回一个包含字符串的list
dir(str)
打印:['__add__','__class__',...,'__subclasshook__','capitalize','casefold',...,'zfill']
len(str)/str._len_()
str.lower()#小写
hasattr(obj,'x')# 有属性'x'吗?
>>> setattr(obj,'y',19)# 设置一个属性'y'
>>> getattr(obj,'y')# 获取属性'y' / obj.y# 获取属性'y'
获取不存在的属性
错误提示:AttributeError
可以传入一个default参数,如果属性不存在,就返回默认值:
>>> getattr(obj,'z',404)# 获取属性'z',如果不存在,返回默认值404
404
实例属性和类属性
>>>class Student(object):
... name ='Student'
>>> s = Student()# 创建实例s
>>>print(s.name)# 打印name属性,因为实例并没有name属性,所以会继续查找class的name属性
Student
>>>print(Student.name)# 打印类的name属性
Student
>>> s.name ='Michael' # 给实例绑定name属性
>>>print(s.name)# 由于实例属性优先级比类属性高,因此,它会屏蔽掉类的name属性
Michael
>>>print(Student.name)# 但是类属性并未消失,用Student.name仍然可以访问
Student
>>>del s.name# 如果删除实例的name属性
>>>print(s.name)# 再次调用s.name,由于实例的name属性没有找到,类的name属性就显示出来了
Student
给实例绑定一个方法
from typesimport MethodType
stu.get_age = MethodType(get_age, stu)#给实例绑定一个方法
stu.get_age(24)
给整个类绑定一个方法
Student.get_age=get_age
IndentationError: unexpected indent #错误使用缩进
classStudent(object):
__slots__ = ('name','age')# 用tuple定义允许绑定的属性名称
使用__slots__要注意,__slots__定义的属性仅对当前类实例起作用,对继承的子类是不起作用的
多继承:通过多重继承,一个子类就可以同时获得多个父类的所有功能。
classBat(Mammal, Flyable):pass
MixIn的目的就是给一个类增加多个功能
class Dog(Mammal, RunnableMixIn, CarnivorousMixIn): #允许多继承
pass
__str__()返回用户看到的字符串,
而__repr__()返回程序开发者看到的字符串,也就是说,__repr__()是为调试服务的。
__iter__
如果一个类想被用于for ... in循环,类似list或tuple那样,就必须实现一个__iter__()方法,该方法返回一个迭代对象,然后,Python的for循环就会不断调用该迭代对象的__next__()方法拿到循环的下一个值,直到遇到StopIteration错误时退出循环。
错误跟踪信息
$ python3 err.py
Traceback (most recent call last): 这是错误的跟踪信息。
File "err.py", line 11, in
main() 调用main()出错了,在代码文件err.py的第11行代码,但原因是第9行:
File "err.py", line 9, in main
bar('0') 调用bar('0')出错了,在代码文件err.py的第9行代码,但原因是第6行
File "err.py", line 6, in bar
return foo(s) * 2 原因是return foo(s) * 2这个语句出错了,但这还不是最终原因,继续往下看:
File "err.py", line 3, in foo
return 10 / int(s) eturn 10 / int(s)这个语句出错了,这是错误产生的源头,因为下面打印了ZeroDivisionError: division by zero 错误类型ZeroDivisionError
单元测试
比如对函数abs(),我们可以编写出以下几个测试用例:
输入正数,比如1、1.2、0.99,期待返回值与输入相同;
输入负数,比如-1、-1.2、-0.99,期待返回值与输入相反;
输入0,期待返回0;
输入非数值类型,比如None、[]、{},期待抛出TypeError。