一、面向对象和面向过程
面向过程编程
核心是过程二字,过程指的是解决问题的步骤,即先干什么再干什么然后干什么。。。
基于该思想编写程序好比在设计一条流水线,是一种机械式的思维方式
优点:复杂的问题流程化、进而简单化
缺点:扩展性差
面向对象编程
核心对象二字,对象是特征与技能的结合体
基于该思想编写程序就好比是在创造一个世界,你就是这个世界的上帝,是一种上帝式的思维方式
优点:可扩展性强
缺点:编程的复杂度要高于面向过程
二、类
''' 类:种类、分类、类别 对象是特征与技能的结合体,类是一系列对象相似的特征与技能的结合体 强调:站的角度不同,总结出的类是截然不同的 在现实世界中:先有的一个个具体存在的对象,然后随着人类文明的发展才有了分类的概念 在程序中:必须先定义类,后调用类来产生对象 现实世界中对对象==》总结出现实世界中的类==》定义程序中的类==》调用类产生程序中的对象 站在老男孩选课系统的角度,先总结现实世界中的老男孩的学生对象(这些对象有相似的特征和技能) 对象1: 特征: 学校='oldboy' 姓名='耗哥' 年龄=18 性别='male' 技能: 选课 对象2: 特征: 学校='oldboy' 姓名='猪哥' 年龄=17 性别='male' 技能: 选课 对象3: 特征: 学校='oldboy' 姓名='帅翔' 年龄=19 性别='female' 技能: 选课 站在老男孩选课系统的角度,先总结现实世界中的老男孩学生类 老男孩学生类: 相似的特征: 学校='oldboy' 相似的技能 选课 ''' #在程序中 #1、先定义类 class OldboyStudent: #先定义类,定义阶段类体代码立刻执行,产生一个类的名称空间,将一系列相似的特征与技能的名字丢到类的名称空间中 school='oldboy' #将student名字丢到类的名称空间中,指向’oldboy'的内存地址 def choose_course(self): #将choose_course名字丢到类的名称空间中,指向函数的内存地址 print('is choosing course') #类体代码会在类定义阶段就立刻执行,会产生一个类的名称空间 # 类的本身其实就是一个容器/名称空间,是用来存放名字的,这是类的用途之一 # print(OldboyStudent.__dict__) #类的字典属性,里面存放了一堆名字 # print(OldboyStudent.__dict__['school']) #数据属性:字典通过key值,就可以取到对应的value值 # print(OldboyStudent.__dict__['choose_course']) #函数属性:字典通过key值,就可以取到对应的value值,value值是是函数的内存地址 # OldboyStudent.__dict__['choose_course']() #内存地址加括号就可以直接调用 # print(OldboyStudent.school) #OldboyStudent.__dict__['school'] #另外一种访问数据属性的方法,直接使用点就可以访问类的名称空间中的名字 # print(OldboyStudent.choose_course) #OldboyStudent.__dict__['choose_course']#另外一种访问函数属性的方法,直接使用点就可以访问类的名称空间中的名字 # OldboyStudent.choose_course(111) #类通过点直接访问的函数属性就是一个普通的函数,要遵循函数的传参的规则 # 类的增删改查 # 增 # OldboyStudent.country='China' #OldboyStudent.__dict__['country']='China' # 改 # OldboyStudent.country='CHINA' #OldboyStudent.__dict__['country']='China' # 删 # del OldboyStudent.school # 查 # print(OldboyStudent.__dict__) #2、后调用类产生对象,调用类的过程,又称为类的实例化,实例化的结果称为类的对象/实例 # stu1=OldboyStudent() # 调用类会得到一个返回值,该返回值就是类的一个具体存在的对象/实例 # stu2=OldboyStudent() # 调用类会得到一个返回值,该返回值就是类的一个具体存在的对象/实例 # stu3=OldboyStudent() # 调用类会得到一个返回值,该返回值就是类的一个具体存在的对象/实例 # 类的实例化过程都发生了哪些事? # 如何在实例化的过程中为对象定制自己独有的特征 # 程序中对象到底是什么,如何使用?
三、为对象定制自己独有的属性
# # 例1 class OldboyStudent: #定义一个类,执行类体代码 school='oldboy' #在类的名称空间产生一个数据属性的名字(特征) def choose_course(self):#在类的名称空间产生一个函数属性的名字(技能) print('is choosing course') stu1=OldboyStudent() #实例化(调用类)产生一个空对象,对象就是一个名称名称空间,就是用来存放一堆自己独有的数据属性的 stu2=OldboyStudent() stu3=OldboyStudent() #对象本质也就是一个名称空间而已,对象名称空间(容器)是用存放对象自己------独有的名字/属性,而 #类中存放的是对象们---------------------------------------------------共有的属性 print(stu1.__dict__) #-----{},通过字典属性,可以查看字典是一个空字典{} print(stu2.__dict__) print(stu3.__dict__) # 问题点:以下产生的三个对象,有相似的特征与技能,首先想到的是将其放到一个函数中,减少代码冗余 stu1.name='耗哥' #本质stu1.__dict__['name']='耗哥' stu1.__dict__['name']='耗哥' #就是往空对象中加入了一个key:value stu1.age=18 stu1.sex='male' print(stu1.name,stu1.age,stu1.sex) #-----耗哥 18 male,通过点就可以访问对象名称空间中的名字,进而拿到与名字绑定的值 print(stu1.__dict__) #-----{'name': '耗哥', 'age': 18, 'sex': 'male'},可以看到对象中独有的数据属性 stu2.name='猪哥' stu2.age=17 stu2.sex='male' stu3.name='帅翔' stu3.age=19 stu3.sex='female' # # 例2--------------------------------------------------------------------------- class OldboyStudent: school='oldboy' def choose_course(self): print('is choosing course') stu1=OldboyStudent() stu2=OldboyStudent() stu3=OldboyStudent() '''在类之外定义类一个函数,将相似的特征封装成了一个函数,不同的对象来调用只需要直接传参就可以了''' def init(obj,x,y,z): #将对象和需要的实参按位置传给形参,将不同的对象传入都可以来调用 obj.name=x obj.age=y obj.sex=z # stu1.name='耗哥' # stu1.age=18 # stu1.sex='male' init(stu1,'耗哥',18,'male') #封装成函数后,上面的三行代码就可以不用写了,减少了代码冗余的目的 # # stu2.name='猪哥' # stu2.age=17 # stu2.sex='male' init(stu2,'诸哥',17,'male') #对象2来调用也只需要,将参传入即可 # stu3.name='帅翔' # stu3.age=19 # stu3.sex='female' init(stu3,'帅翔',19,'female') #对象3来调用也只需要,将参传入即可 print(stu1.__dict__) #--------{'name': '耗哥', 'age': 18, 'sex': 'male'} print(stu2.__dict__) #--------{'name': '诸哥', 'age': 17, 'sex': 'male'} print(stu3.__dict__) #--------{'name': '帅翔', 'age': 19, 'sex': 'female'} # 例3 # 在类体内定义一个__init__函数,将其名字存放在类的名称空间中,这样可以被所有调用类产生的对象引用 class OldboyStudent: school='oldboy' def __init__(obj, x, y, z): #会在调用类时自动触发,(调用类产生对象,将对象自动传入给obj,而此时的obj不在仅仅是一个形参,而是一个对象,对象就是一个容器,容器内存放很多名字,都可以被引用) obj.name = x #stu1.name='耗哥' (通过点拿到对象名称空间中的name) obj.age = y #stu1.age=18 #给对象添加独有的数据属性 obj.sex = z #stu1.sex='male' def choose_course(self): #choose_course是存放在类的名称空间中的,哪个对象来调用就会将其绑定给谁,并将对象当做第一个参数传入(精髓在与对象就是一个名称空间,这样对象中的名称空间的名字都可以被引用) print('is choosing course') #调用类时发生两件事 #1、创造一个空对象stu1 #2、自动触发类中__init__功能的执行,将stu1以及调用类括号内的参数一同传入 stu1=OldboyStudent('耗哥',18,'male') #本质就是:OldboyStudent.__init__(stu1,'耗哥',18,'male')-----类点一个类体代码内的函数属性,就是跟访问一个普通函数没有任何区别,按照函数的属性给其传参即可 stu2=OldboyStudent('猪哥',17,'male') stu3=OldboyStudent('帅翔',19,'female') # print(stu1.__dict__) #------{'name': '耗哥', 'age': 18, 'sex': 'male'} print(stu2.__dict__) #------{'name': '猪哥', 'age': 17, 'sex': 'male'} print(stu3.__dict__) #------{'name': '帅翔', 'age': 19, 'sex': 'female'}
配图
四、属性查找
class OldboyStudent: #------------------重点:类体代码产生的所用名字都放在类的名称空间中,这样可以被共享给所有的对象用,对象通过点就可以引用 school='oldboy' count=0 #这个count是放在类的名称空间中,可以给调用类产生的不同的对象引用 def __init__(self, x, y, z): #会在调用类时自动触发 self.name = x #stu1.name='耗哥' self.age = y #stu1.age=18 self.sex = z #stu1.sex='male' OldboyStudent.count+=1 #每调用一次类产生一个对象,count就加一,调用单词就加了三次 def choose_course(self): print('is choosing course') # 先从对象自己的名称空间找,没有则去类中找,如果类也没有则报错 stu1=OldboyStudent('耗哥',18,'male') stu2=OldboyStudent('猪哥',17,'male') stu3=OldboyStudent('帅翔',19,'female') print(OldboyStudent.count) #类直接点访问类名称空间中的名字 print(stu1.count) #-----类被调用了三次所以结果是3:对象点名字,会先从自己的名称空间找,没找到到自己类的名称空间找 print(stu2.count) print(stu3.count)
五、绑定方法
class OldboyStudent: school='oldboy' #-------------------------类中定义的数据属性是共享给所有对象用的 def __init__(self, x, y, z): #会在调用类时自动触发--------类中定义的所有函数属性是共享给所有对象用的 self.name = x #stu1.name='耗哥'======》stu1.__dict__['name']='耗哥' self.age = y #stu1.age=18 self.sex = z #stu1.sex='male' def choose_course(self,x): #-------------------------类中定义的所有函数属性是共享给所有对象用的,调用类会将对象自动传入,就可以应用对象下的数据属性 print('%s is choosing course' %self.name)#-----------通过点引用了对象下的name数据属性 # def func(): #类中定义的函数会默认传入一个self,如果不写调用类时会自动将产生的对象传入,此时就会报错,TypeError: func() takes 0 positional arguments but 1 was given # pass # 类名称空间中定义的数据属性和函数属性都是共享给所有对象用的 # 对象名称空间中定义的只有数据属性,而且是对象所独有的数据属性 stu1=OldboyStudent('耗哥',18,'male') #调用类,会将对象自动传入给self,参数会按位置传__init__函数的其他形参 stu2=OldboyStudent('猪哥',17,'male') stu3=OldboyStudent('帅翔',19,'female') print(stu1) # print(stu1.name) #访问对象下的数据属性 # print(stu1.school) # 类中定义的函数是类的函数属性,类可以使用,但使用的就是一个普通的函数而已,意味着需要完全遵循函数的参数规则,该传几个值就传几个 # print(OldboyStudent.choose_course) #打印结构是一个函数的内存地址,所以类通过点使用类中定义的函数,跟使用一个普通函数一样,需要参数传参即可 # OldboyStudent.choose_course(123) # 类中定义的函数是共享给所有对象的,对象也可以使用,而且是绑定给对象用的, #绑定的效果:绑定给谁,就应该由谁来调用,谁来调用就会将谁当作第一个参数自动传入(自动传入给self) # print(id(stu1.choose_course)) #2176273098184(每次运行同一个对象得到的id是不一样的,因为每次都要重新申请一个内存地址) # print(id(stu2.choose_course)) #2176273098184(将类的函数属性绑定给对象) # print(id(stu3.choose_course)) #2176273098184 #三个不同的对象应用了类的名称空间的函数属性,指向的是同一个内存地址,所以三者的id值一样的 # print(id(OldboyStudent.choose_course)) # # print(id(stu1.school)) #将类的数据属性绑定给对象 # print(id(stu2.school)) # print(id(stu3.school)) # # print(id(stu1.name),id(stu2.name),id(stu3.name)) # stu1.choose_course(1) #绑定给对象用的,所以会将对象当作第一个参数传入,即此时我们只需要再传入一个参数就可以了,并不是self没有被传参 # stu2.choose_course(2) # stu3.choose_course(3) # stu1.func() # 重点: # 补充:类中定义的函数,类确实可以使用,但其实类定义的函数大多情况下都是绑定给对象用的,所以在类中定义的函数都应该自带一个参数self
六、类即类型
#在python3中统一了类与类型的概念,类就是类型 class OldboyStudent: school='oldboy' def __init__(self, x, y, z): #会在调用类时自动触发 self.name = x #stu1.name='耗哥' self.age = y #stu1.age=18 self.sex = z #stu1.sex='male' def choose_course(self,x): print('%s is choosing course' %self.name) stu1=OldboyStudent('耗哥',18,'male') # stu1.choose_course(1) #OldboyStudent.choose_course(stu1,1) # OldboyStudent.choose_course(stu1,1) l=[1,2,3] #l=list([1,2,3]) #list即列表类型也即列表类,l就是调用列表类加括号产生的对象 print(type(l)) #<class 'list'> # l.append(4) #list.append(l,4) #类可以访问类中定义的函数属性,对象也可以访问类中定义的函数属性 list.append(l,4) # print(l)
七、小结
''' 重点 # 对象是一个高度整合的产物,整合数据与专门操作该数据的方法(绑定方法) class Foo: def __init__(self,host,port,db,chartset): self.host=host self.port=port self.db=db self.charset=charset def exc1(self,sql): conn = connect(self.host, self.port, self.db, self.charset) conn.execute(sql) return xxx def exc2(self,proc_name) conn = connect(self.host, self.port, self.db, self.charsett) conn.call_proc(sql) return xxx class Bar: def __init__(self,x,y,z,a,b,c): self.x=x self.y=y self.z=z self.a=a self.b=b self.c=c def exc3(self,xxx): pass def exc4(self,yyy) pass obj1=Foo('1.1.1.1',3306,'db1','utf-8') obj1.exc1('select * from t1') obj1.exc1('select * from t2') obj1.exc1('select * from t3') obj1.exc1('select * from t4') obj2=Foo('1.1.1.2',3306,'db1','utf-8') obj2.exc1('select * from t4') '''
八、继承与派生
''' 1、什么是继承 继承是一种新建类的方式,新建的类称为子类,被继承的类称为父类 继承的特性是:子类会遗传父类的属性 强调:继承是类与类之间的关系 2、为什么用继承 继承的好处就是可以减少代码的冗余 3、如何用继承 在python中支持一个类同时继承多个父类 在python3中 如果一个类没有继承任何类,那默认继承object类 在python2中: 如果一个类没有继承任何类,不会继承object类 新式类 但凡继承了object的类以及该类的子类,都是新式类 经典类 没有继承object的类以及该类的子类,都是经典类 在python3中都是新式类,只有在python2中才区别新式类与经典类 新式类vs经典类? ''' class Parent1(object): pass class Parent2(object): pass class Sub1(Parent1,Parent2): pass print(Sub1.__bases__) #bases是查看类的所有父类(<class '__main__.Parent1'>, <class '__main__.Parent2'>) print(Parent1.__bases__) #(<class 'object'>,)-----没有父类,默认父类是object print(Parent2.__bases__) #(<class 'object'>,)-----没有父类,默认父类是object
九、继承的应用
#派生:子类中新定义的属性,子类在使用时始终以自己的为准 class OldboyPeople: #父类----------------将类与类相似的数据属性和函数属性放在父类中,这样可以减少代码冗余 school = 'oldboy' def __init__(self,name,age,sex): #定制对象自己独有的数据属性,哪个对象来调用,哪个对象就具有该数据属性,我们只需要给其添加属性值即可 self.name = name #tea1.name='egon' self.age = age #tea1.age=18 self.sex = sex #tea1.sex='male' class OldboyStudent(OldboyPeople):#子类-------子类内的就是派生类,就是除父类之外的自己独有的数据属性和函数属性 def choose_course(self): print('%s is choosing course' %self.name) class OldboyTeacher(OldboyPeople):#子类-------子类内的就是派生类,就是除父类之外的自己独有的数据属性和函数属性 # tea1,'egon',18,'male',10 def __init__(self,name,age,sex,level): # self.name=name #左边name是属性,右边name是变量 # self.age=age # self.sex=sex OldboyPeople.__init__(self,name,age,sex)#(实现代码的重用,实现代码的冗余)类点一个函数属性和引用一个普通的函数没有什么区别,按照位置传参就可以了,指名道姓,与继承无关 self.level=level def score(self,stu_obj,num):#传入要给的学生对象,和要改的分数,stu1是一个高度整合的产物,拿到self不仅拿到对象独有的数据属性还有公有的属性,还有处理数据的所有方法 print('%s is scoring' %self.name) stu_obj.score=num #这样就可以拿到学生对象的名称空间,将其数据属性score进行修改,实现不同类的数据属性进行交互 stu1=OldboyStudent('耗哥',18,'male') tea1=OldboyTeacher('egon',18,'male',10) #对象查找属性的顺序:对象自己-》对象的类-》父类-》父类。。。 # print(stu1.school) # print(tea1.school) # print(stu1.__dict__) # print(tea1.__dict__) tea1.score(stu1,99) #----egon is scoring,对象绑定到方法,会将对象自动传入,再将其他参数传入,再有该对象来调用该函数,执行函数体代码 # print(stu1.__dict__) #{'name': '耗哥', 'age': 18, 'sex': 'male', 'score': 99} # 在子类派生出的新功能中重用父类功能的方式有两种: #1、指名道姓访问某一个类的函数:该方式与继承无关
# class Foo: # def f1(self): # print('Foo.f1') # # def f2(self): # print('Foo.f2') # self.f1() # # class Bar(Foo): # def f1(self): # print('Bar.f1') # # #对象查找属性的顺序:对象自己-》对象的类-》父类-》父类。。。 # obj=Bar() # obj.f2() # ''' # Foo.f2 # Bar.f1 # '''