内容概要
内容详细
对象的概念
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| '面对对象'的核心就是'对象'二字,而对象的精髓在于'整合'
所有的程序都是由'数据' + '功能'组成,因而编写程序的本质就是:定义出一系列的数据,然后定义出一系列的功能来对数据进行操作
在学习'对象'之前,程序中的数据与功能是分离开的,例如:
name='lili' age=18 sex='female'
def tell_info(name,age,sex): print('<%s:%s:%s>' %(name,age,sex))
tell_info(name,age,sex)
|
1 2 3 4 5 6 7 8 9 10
| ========================================================================================
在学习了'对象'之后,我们就有了一个容器,该容器可以盛放数据与功能,所以我们可以说:对象是把数据与功能整合到一起的产物,或者说'对象'就是一个盛放数据与功能的容器
如果: 把'数据'比喻为'睫毛膏'、'眼影'、'唇彩'等化妆所需要的原材料 把'功能'比喻为'眼线笔'、'眉笔'等化妆所需要的工具 那么: '对象'就是一个'彩妆盒','彩妆盒'可以把'原材料'与'工具'都装到一起
|

1 2 3 4 5 6 7
| 如果我们把'化妆'比喻为要执行的业务逻辑,此时只需要拿来一样东西即可,那就是彩妆盒,因为彩妆盒里整合了'化妆'所需的所有原材料与功能
这比起分别拿来原材料与功能才能执行,要方便的多
在了解了对象的基本概念之后,理解面向对象的编程方式就相对简单很多了,面向对象编程就是要造出一个个的对象,把原本分散开的相关数据'整合'到一个个的对象里,这么做既方便使用,也可以提高程序的解耦合程度,进而提升了程序的可扩展性
强调:软件质量属性包含很多方面,面向对象解决的仅仅是扩展性问题
|

类与对象
1 2 3 4 5
| 类 ---> 类别/种类:是免息那个对象分析和设计的基石 如果多个对象有相似的数据与功能,那么该多个对象就属于同一种类 有了类的好处: 可以把同一类的对象相同的数据与功能存放到类里,而无需每个对象都重复存一份,这样每个对象里只需存自己独有的数据即可,极大地节省了空间 所以 ---> 如果说对象是用来存放数据与功能的容器,那么类则是用来存放多个对象相同的数据与功能的容器
|

1 2 3
| 综上所述: 1 虽然我们是先介绍对象,后介绍类,但是:'在程序中,必须要事先定义类,然后再调用类产生对象(调用类类拿到的返回值就是对象)' 2 产生对象的类与对象之间存在关联,这种关联指的是:对象可以访问到类中共有的数据与功能,所以类中的内容仍然是属于对象的,类只不过是一种节省空间,减少代码冗余的机制,面对对象编程最终的核心仍然是去使用对象
|
面对对象编程
类的定义与实例化
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
| 我们以开发一个清华大学的选课系统为例,来简单介绍基于面向对象的思想如何编写程序
面向对象的基本思路就是把程序中要用到的、相关联的数据与功能整合到对象里,然后再去使用,但程序中要用到的数据以及功能那么多,如何找到相关连的呢?我需要先提取选课系统里的角色:学生、老师、课程等,然后显而易见的是:学生有学生相关的数据于功能,老师有老师相关的数据与功能,我们单以学生为例,
学校 名字 年龄 性别
选课 详细的
数据: 学校=清华大学 姓名=李建刚 性别=男 年龄=28 功能: 选课
数据: 学校=清华大学 姓名=王大力 性别=女 年龄=18 功能: 选课
数据: 学校=清华大学 姓名=牛嗷嗷 性别=男 年龄=38 功能: 选课 我们可以总结出一个学生类,用来存放学生们相同的数据与功能
相同的特征: 学校=清华大学 相同的功能: 选课
|
1 2 3 4 5 6 7 8
| 基于上述分析的结果,我们接下来需要做的就是在程序中定义出类,然后调用类产生对象
class Student:
school='清华大学'
def choose(self): print('%s is choosing a course' %self.name)
|
1 2 3 4
| 类体最常见的是变量的定义和函数的定义,但其实类体可以包含任意Python代码,类体的代码在类定义阶段就会执行,因而会产生新的名称空间用来存放类中定义的名字,可以打印Student.__dict__来查看类这个容器内盛放的东西
>>> print(Student.__dict__) {..., 'school': '清华大学', 'choose': <function Student.choose at 0x1018a2950>, ...}
|
1 2 3 4 5 6
| 调用类的过程称为将类实例化,拿到的返回值就是程序中的对象,或称为一个实例
>>> stu1=Student() >>> stu2=Student() >>> stu3=Student()
|
1 2 3 4 5 6 7 8 9 10 11 12 13
| 如此stu1、stu2、stu3全都一样了(只有类中共有的内容,而没有各自独有的数据),想在实例化的过程中就为三位学生定制各自独有的数据:姓名,性别,年龄,需要我们在类内部新增一个__init__方法,如下
class Student: school='清华大学'
def __init__(self,name,sex,age): self.name=name self.sex=sex self.age=age
def choose(self): print('%s is choosing a course' %self.name)
|
1 2 3 4 5
| 然后我们重新实例出三位学生
>>> stu1=Student('李建刚','男',28) >>> stu2=Student('王大力','女',18) >>> stu3=Student('牛嗷嗷','男',38)
|
1 2 3 4 5 6
| 单拿stu1的产生过程来分析,调用类会先产生一个空对象stu1,然后将stu1连同调用类时括号内的参数一起传给Student.__init__(stu1,’李建刚’,’男’,28)
def __init__(self, name, sex, age): self.name = name self.sex = sex self.age = age
|
1 2 3 4
| 会产生对象的名称空间,同样可以用__dict__查看
>>> stu1.__dict__ {'name': '李建刚', 'sex': '男', 'age': 28}
|
至此,我们造出了三个对象与一个类,对象存放各自独有的数据,类中存放对象们共有的内容

存的目的是为了用,那么如何访问对象或者类中存放的内容呢?
属性访问
类属性与对象属性
1 2 3 4 5 6 7
| 在类中定义的名字,都是类的属性,细说的话,类有两种属性:数据属性和函数属性,可以通过__dict__访问属性的值,比如Student.__dict__[‘school’],但Python提供了专门的属性访问语法
>>> Student.school '清华大学' >>> Student.choose <function Student.choose at 0x1018a2950>
|
1 2 3 4 5 6 7
| 操作对象的属性也是一样
>>> stu1.name '李建刚' >>> stu1.course=’python’ >>> stu1.age=38 >>> del obj1.course
|
属性查找与绑定方法
1 2 3 4 5 6 7 8 9 10
| 对象的名称空间里只存放着对象独有的属性,而对象们相似的属性是存放于类中的。对象在访问属性时,会优先从对象本身的__dict__中查找,未找到,则去类的__dict__中查找
1、类中定义的变量是类的数据属性,是共享给所有对象用的,指向相同的内存地址
print(id(Student.school))
print(id(stu1.school)) print(id(stu2.school)) print(id(stu3.school))
|
1 2 3 4 5 6
| 2、类中定义的函数是类的函数属性,类可以使用,但必须遵循函数的参数规则,有几个参数需要传几个参数
Student.choose(stu1) Student.choose(stu2) Student.choose(stu3)
|
1 2 3 4 5 6 7
| 但其实类中定义的函数主要是给对象使用的,而且是绑定给对象的,虽然所有对象指向的都是相同的功能,但是绑定到不同的对象就是不同的绑定方法,内存地址各不相同
print(id(Student.choose))
print(id(stu1.choose)) print(id(stu2.choose)) print(id(stu3.choose))
|
1 2 3 4 5 6
| 绑定到对象的方法特殊之处在于,绑定给谁就应该由谁来调用,谁来调用,就会将’谁’本身当做第一个参数自动传入(方法__init__也是一样的道理)
stu1.choose() stu2.choose() stu3.choose()
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
| 绑定到不同对象的choose技能,虽然都是选课,但李建刚选的课,不会选给王大力,这正是”绑定“二字的精髓所在。
Python中一切皆为对象,且Python3中类与类型是一个概念,因而绑定方法我们早就接触过
>>> list <class 'list'>
#实例化的到3个对象l1,l2,l3 >>> l1=list([1,2,3]) >>> l2=list(['a','b','c']) >>> l3=list(['x','y'])
#三个对象都有绑定方法append,是相同的功能,但内存地址不同 >>> l1.append <built-in method append of list object at 0x10b482b48> >>> l2.append <built-in method append of list object at 0x10b482b88> >>> l3.append <built-in method append of list object at 0x10b482bc8>
#操作绑定方法l1.append(4),就是在往l1添加4,绝对不会将4添加到l2或l3 >>> l1.append(4) #等同于list.append(l1,4) >>> l1 [1,2,3,4] >>> l2 ['a','b','c'] >>> l3 ['x','y']
|
总结
在上述介绍类与对象的使用过程中,我们更多的是站在底层原理的角度去介绍类与对象之间的关联关系,如果只是站在使用的角度,我们无需考虑语法”对象.属性”中”属性”到底源自于哪里,只需要知道是通过对象获取到的就可以了,所以说,对象是一个高度整合的产物,有了对象,我们只需要使用”对象.xxx“的语法就可以得到跟这个对象相关的所有数据与功能,十分方便且解耦合程度极高。