一、类
定义一个类,例子如下:
1 class Turtle: #定义一个名为Turtle的类,Python中类型约定以大写字母开头 2 #属性 3 color='green' 4 weight=10 5 legs=4 6 shell=True 7 mouth='大嘴' 8 9 #方法 10 def climb(self): 11 print('我正在很努力的向前爬……') 12 def run(self): 13 print('我正在飞快的向前跑……') 14 def bite(self): 15 print('咬死你咬死你!') 16 def eat(self): 17 print('邮的吃,真满足~') 18 def sleep(self): 19 print('困了,睡了,晚安,Zzzzz')
【属性+方法-->实例化对象】
定义一个对象,调用类中方法:
1 >>> tt=Turtle() 2 >>> tt.climb #未调用成功,内存中调用一个实例对象,系统会自动给释放 3 <bound method Turtle.climb of <__main__.Turtle object at 0x020105B0>> 4 >>> tt.climb() 5 我正在很努力的向上爬
二、
1.封装:信息隐蔽技术
1 >>> list1=[2,1,7,5,3] 2 >>> list1.sort() 3 >>> list1 4 [1, 2, 3, 5, 7] 5 >>> list1.append(9) #append方法我们知道应用后可以达到什么效果但是却不知道它的实现原理,那么这样的方法就是封装起来的 6 >>> list1 7 [1, 2, 3, 5, 7, 9]
2.继承
1 #创建一个list类型的类 2 >>> class MyList(list): 3 pass #pass的意思,pass是一个占位符,不做任何事情,此类是一个list(列表) 4 #创建一个list2对象 5 >>> list2=MyList() 6 >>> list2.append(5) 7 >>> list2.append(3) 8 >>> list2.append(7) 9 >>> list2 10 [5, 3, 7] 11 >>> list2.sort() 12 >>> list2 13 [3, 5, 7] #这里实现了排序方法,所以list2继承了Mylist()中的list的方法
3.多态:不同对象对同一方法相应不同的行动
【概念理解】奔跑:老鼠、袋鼠、乌龟对于同一个跑的动作都是表现不同的行为的。
1 >>> class A: 2 def fun(self): 3 print('我是小A') 4 5 >>> class B: 6 def fun(self): 7 print('我是小B') 8 9 10 >>> a=A() 11 >>> b=B() 12 >>> a.fun() 13 我是小A 14 >>> b.fun() 15 我是小B
4.self:相当于不同对象的门牌号。python中调用时候第一个写self是约定俗成的要求(虽然通过同一个类创建的出不同对象,但是通过self可以区别各个对象。)
调用对象原理:来源于同一个属性和方法,当一个对象被调用,对象会把自身作为第一个参数传给self,接收到self(self作为对象的标志,区别不同对象),python通过对象传过来的self知道是哪个对象在调用方法。
1 >>> class Ball: 2 def setName(self,name): #self相当于字典中的key,name相当于字典中的值,所以括号中实际上只传name的值即可。 3 self.name=name 4 def kick(self): #调用setName()方法中的self,即取出self.name的值 5 print('我叫%s,该死的,谁踢我...'%self.name) 6 7 8 >>> a=Ball() 9 >>> a.setName('球A') 10 >>> b=Ball() 11 >>> b.setName('球B') 12 >>> c=Ball() 13 >>> c.setName('土豆') 14 >>> a.kick() 15 我叫球A,该死的,谁踢我... 16 >>> c.kick() 17 我叫土豆,该死的,谁踢我...
5.Python的魔法方法:
1)__init__(self):构造方法,实例化一个对象时,方法在对象被创建时候自动被调用。不用给对象中各方法中变量分别赋值,只初始化一次赋值即可。例子代码如下:
1 >>> class Ball: 2 def __init__(self,name): #应用__init__函数 3 self.name=name 4 def kick(self): 5 print('我叫%s,该死的,谁踢我...'%self.name) 6 7 8 >>> b=Ball('土豆')#初始化赋值 9 >>> b.kick() 10 我叫土豆,该死的,谁踢我...
6.共有和私有
在属性前面加双下划线‘__’可以使属性变成私有属性(原理:其实是Python动了手脚,python会将加双下划线的变量改名为‘_类名__变量名’,所以仍旧可以通过方法进行访问,所以Python类是没有权限控制的,所以实际上是伪私有),那么对象在进行方法调用的时候不可以直接调用该私有属性。可以通过在方法中再利用调用函数来调用,然后外部对象再调用此方法,从而间接的调用出此属性。(下方有举例)
1 #共有属性: 2 >>> class Person: 3 name='小甲鱼' 4 5 >>> p=Person() 6 >>> p.name 7 '小甲鱼' 8 #私有属性,调用报错 9 >>> class Person: 10 __name='小甲鱼' 11 12 >>> p=Person() 13 >>> p.__name 14 Traceback (most recent call last): 15 File "<pyshell#75>", line 1, in <module> 16 p.__name 17 AttributeError: 'Person' object has no attribute '__name' 18 #私有属性中写入内部调用方法实现间接调用私有方法
>>> class Person: __name='小甲鱼' def getName(self): return self.__name >>> p=Person() >>> p.getName() '小甲鱼'
1 #了解Python私有属性改名原理后进行再次访问尝试 2 >>> class Person: 3 __name='小甲鱼' 4 5 6 >>> p=Person() 7 >>> p._Person__name 8 '小甲鱼'
二、继承
1.概念
1)父类:又名超类、基类、父类
2)子类:
2.语法:class 子类名(父类名):
1 >>> class Parent: 2 def hello(self): 3 print('正在调用父类的方法...') 4 5 6 >>> class Child(Parent): 7 pass #类中没有内容可以用pass代替 8 9 >>> p=Parent() 10 >>> p.hello() 11 正在调用父类的方法... 12 >>> c=Child() 13 >>> c.hello() 14 正在调用父类的方法... 15 #如果子类中定义与父类同名的方法或属性,则会自动覆盖父类对应的方法或属性
>>> class Child(Parent): def hello(self): print('正在调用子类的方法...') >>> c=Child() >>> c.hello() 正在调用子类的方法... >>> p.hello <bound method Parent.hello of <__main__.Parent object at 0x0221FEB0>> >>> p.hello() 正在调用父类的方法...
3.子类中如果重写了父类,那么可以用以下两种方法来调用父类中方法:
1)调用未绑定的父类方法
2)使用super函数
1 import random as r 2 class Fish: 3 def __init__(self): 4 self.x=r.randint(0,10) #x轴坐标 5 self.y=r.randint(0,10) 6 7 def move(self): 8 self.x-=1 #鱼每次都向左游 9 print('我的位置是:',self.x,self.y) 10 11 class Goldfish(Fish): 12 pass 13 class Carp(Fish): 14 pass 15 class Salmon(Fish): 16 pass 17 class Shark(Fish): 18 def __init__(self):#子类重写了父类的此方法,所以有了下面一句话(Fish.__init__(self))来调用父类 19 Fish.__init__(self) #1.此处的self为子类shark的self而非父类的self所以称之为未绑定的父类方法
# 2.(Fish.__init__(self))也可以用super方法来写(super().__init__()),super方法其实更为常见,如果继承多个父类,那么super方法相对更简便 20 self.hungry=True 21 22 def eat(self): 23 if self.hungry: 24 print('吃货的梦想就是天天有鱼吃') 25 self.hungry=False 26 else: 27 print('太撑了,吃不下了')
代码F5后运行结果:
1 >>> shark=Shark() 2 >>> shark.eat()#调用shark中的eat方法 3 吃货的梦想就是天天有鱼吃 4 >>> shark.move()#调用shark对象中的move方法 5 我的位置是: 1 2 6 >>> shark.move() 7 我的位置是: 0 2 8 >>> shark.move() 9 我的位置是: -1 2 10 >>> Fish.__init__(shark)#调用未绑定父类方法 11 >>> shark.move() 12 我的位置是: 2 10
4.多重继承:
1)语法class 子类(父类1,父类2,父类3):
2)实例:
1 >>> class Base1: 2 def foo1(self): 3 print('我是foo1,我为Base1代言...') 4 5 6 >>> class Base2: 7 def foo2(self): 8 print('我是foo2,我为Base2代言...') 9 10 11 >>> class C(Base1,Base2): 12 pass 13 14 >>> c=C() 15 >>> c.foo1() 16 我是foo1,我为Base1代言... 17 >>> c.foo2() 18 我是foo2,我为Base2代言...
属性名同方法名相同,属性会覆盖方法
5.绑定
什么是绑定:Python严格要求方法需要有实例才能被调用,这种限制其实就是Python所谓的绑定概念。
1 >>> class BB: 2 def printBB(): 3 print('no zuo no die') 4 5 >>> BB.printBB() 6 no zuo no die 7 >>> bb=BB() 8 >>> bb.printBB() #这句实际上等价于bb.printBB(bb)传进去bb,但是实际上程序中不识别参数,所以无法调用 9 Traceback (most recent call last): 10 File "<pyshell#6>", line 1, in <module> 11 bb.printBB() 12 TypeError: printBB() takes 0 positional arguments but 1 was given 13 >>> class CC: 14 def setXY(self,x,y): 15 self.x=x 16 self.y=y 17 def printXY(self): 18 print(self.x,self.y) 19 20 21 >>> dd=CC() #实例化一个对象dd 22 >>> dd.__dict__ 23 {} #目前是一个空的字典 24 >>> CC.__dict__ 25 mappingproxy({'__module__': '__main__', '__weakref__': <attribute '__weakref__' of 'CC' objects>, 'printXY': <function CC.printXY at 0x0224DD20>, '__dict__': <attribute '__dict__' of 'CC' objects>, 'setXY': <function CC.setXY at 0x0224DCD8>, '__doc__': None}) 26 >>> dd.setXY 27 <bound method CC.setXY of <__main__.CC object at 0x02240E50>> 28 >>> dd.setXY(4,5) #dd.setXY(dd,4,5)相当于x设置为4,y设置为5,相当于dd对象绑定了4,5。4,5,传入了dd的空间 29 >>> dd.__dict__ 30 {'y': 5, 'x': 4} 31 >>> CC.__dict__ #由于dd已经绑定了4,5,所以CC不会调用到4,5 32 mappingproxy({'__module__': '__main__', '__weakref__': <attribute '__weakref__' of 'CC' objects>, 'printXY': <function CC.printXY at 0x0224DD20>, '__dict__': <attribute '__dict__' of 'CC' objects>, 'setXY': <function CC.setXY at 0x0224DCD8>, '__doc__': None}) 33 >>> dd.setXY(dd,4,5) 34 #删除CC类对象,后看dd是否存在 35 >>> del CC 36 >>> ee=CC() 37 Traceback (most recent call last): 38 File "<pyshell#23>", line 1, in <module> 39 ee=CC() 40 NameError: name 'CC' is not defined 41 >>> dd.printXY() #删除CChou,还可以调用dd的实例对象,类中定义的属性是静态,类被删除,但是其中的属性依旧存在,所以dd仍存在。所以定义类的时候里面尽量定义self.的写法。 42 4 5