类和对象
面向对象编程的2个非常重要的概念:
类和对象
对象是面向对象编程的核心,在使用对象的过程中,为了将具有共同特征和行为的一组对象抽象定义,提出了另外一个新的概念——类
类就相当于制造飞机时的图纸,用它来进行创建的飞机就相当于对象
类是
抽象
的,在使用的时候通常会找到这个类的一个具体的存在,使用这个具体的存在。
一个类可以找到多个对象
定义一个类,格式如下:
class 类名:
方法列表
demo:定义一个Hero类 # class Hero: # 经典类(旧式类)定义形式
class Hero(object): # 新式类定义形式
def info(self):
print("英雄各有见,何必问出处。")
说明:
定义类时有2种形式:新式类和经典类,上面代码中的Hero为新式类,注释部分则为经典类;
object 是Python 里所有类的最顶级父类;
类名 的命名规则按照"大驼峰命名法";
info 是一个实例方法,第一个参数一般是self,表示实例对象本身,当然了可以将self换为其它的名字,其作用是一个变量 这个变量指向了实例对象
对象创建并添加属性后,能否在类的实例方法里获取这些属性呢?如果可以的话,应该通过什么方式?
答案:
在方法内通过self获取对象属性
魔法方法__init__
Python 的类里提供的,两个下划线开始,两个下划线结束的方法,就是魔法方法,__init__()就是一个魔法方法,通常用来做属性初始化 或 赋值 操作(作用)。
# 如果类面没有写__init__方法,Python会自动创建,但是不执行任何操作,
# 如果为了能够在完成自己想要的功能,可以自己定义__init__方法,
# 所以一个类里无论自己是否编写__init__方法 一定有__init__方法。
通过一个类,可以创建多个对象,就好比 通过一个模具创建多个实体一样
__init__(self)中,默认有1个参数名字为self,如果在创建对象时传递了2个实参,那么__init__(self)中出了self作为第一个形参外还需要2个形参,例如__init__(self,x,y)
注意:
1). 在类内部获取 属性 和 实例方法,通过self获取;
2). 在类外部获取 属性 和 实例方法,通过对象名获取。
3). 如果一个类有多个对象,每个对象的属性是各自保存的,都有各自独立的地址;
4). 但是实例方法是所有对象共享的,只占用一份内存空间。类会通过self来判断是哪个对象调用了实例方法。
#自己定义的类 包含info方法和赋值(添加数据)
class Animal(object):
def __init__(self,name,sex,ms,num):
self.name=name #self获取值
self.ser=sex
self.ms=ms
self.num=num
print("小狗姓名是%s"%self.name)
print("小狗性别是%s" % self.ser)
print("小狗毛色是%s" % self.ms)
print("小狗腿的数量是%s" % self.num)
def run(self):
print("狗在跑")
在python中方法名如果是__xxxx__()的,那么就有特殊的功能,因此叫做“魔法”方法
当使用print输出对象的时候,默认打印对象的内存地址。如果类定义了__str__(self)方法,那么就会打印从在这个方法中 return 的数据
__str__方法通常返回一个字符串,作为这个对象的描述信息
def __str__(self):
return "姓名是%s性别是%s"%(self.name,self.ser)
#call 如果一个类中定义了__call__方法,对象后面加括号,会触发执行__call__方法
def __call__(self, *args, **kwargs):
print("姓名是aa")
a=Animal("旺财","公","金黄",4)
print(a)
a.run()#获取属性
a()
继承
程序中的继承在程序中,继承描述的是多个类之间的所属关系。
如果一个类A里面的属性和方法可以复用,则可以通过继承的方式,传递到类B里。
那么类A就是基类,也叫做父类;类B就是派生类,也叫做子类
单继承:
虽然子类没有定义__init__方法初始化属性,也没有定义实例方法,但是父类有。所以只要创建子类的对象,就默认执行了那个继承过来的__init__方法
总结:
子类在继承的时候,在定义类时,小括号()中为父类的名字
父类的属性、方法,会被继承给子类
多继承:
多继承可以继承多个父类,也继承了所有父类的属性和方法
注意:如果多个父类中有同名的 属性和方法,则默认使用第一个父类的属性和方法(根据类的魔法属性mro的顺序来查找)
多个父类中,不重名的属性和方法,不会有任何影响。
class Master(object):
def __init__(self):
self.kongfu="古法煎饼果子配方" #实例变量,属性
def make_cake(self): #实例方法,方法
print("[古法]按照<%s>制作了一份煎饼国子",self.kongfu)
def dayandai(self):
print("大烟袋")
class school(object):
def __init__(self):
self.kongfu = "现代煎饼果子配方"
def make_cake(self): #实例方法,方法
print("[现代]按照<%s>制作了一份煎饼果子"%self.kongfu)
def xiaodayandai(self):
print("小烟袋")
class Prence(school,Master):#继承多个父类
def __init__(self):
self.kongfu="猫氏煎饼果子配方"
def make_cake(self):
print("按照<%s>制作了一份煎饼果子"%self.kongfu)
class Prentice(Master):
#子类可以继承父类的所有属性和方法,哪怕自己没有的属性
pass
damao=Prence()#创建子类实例对象
print(Prence.__mro__)
print(damao.kongfu)#子类对象可以直接使用父类的属性
damao.make_cake()#子类对象可以直接使用父类的方法
damao.dayandai()
damao.xiaodayandai()