Python的面向对象编程方式学习笔记


Posted in Python onJuly 12, 2016

类与实例
类与实例相互关联着:类是对象的定义,而实例是“真正的实物”,它存放了类中所定义的对象的具体信息。

下面的示例展示了如何创建一个类:

class MyNewObjectType(bases):
   ''' 创建 MyNewObjectType 类'''
   class_suite

关键字是 class,紧接着一个类名。随后是定义类的类代码。这里通常由各种各样的定义和声明组成。新式类和经典类声明的最大不同在于,所有新式类必须继承至少一个父类,参数 bases 可以是一个(单继承)或多个(多重继承)用于继承的父类。

创建一个实例的过程称作实例化,过程如下(注意:没有使用 new 关键字):

myFirstObject = MyNewObjectType()

类名使用我们所熟悉的函数操作符(()),以“函数调用”的形式出现。然后你通常会把这个新建的实例赋给一个变量。赋值在语法上不是必须的,但如果你没有把这个实例保存到一个变量中,它就没用了,会被自动垃圾收集器回收,因为任何引用指向这个实例。这样,你刚刚所做的一切,就是为那个实例分配了一块内存,随即又释放了它。

类既可以很简单,也可以很复杂,这全凭你的需要。最简单的情况,类仅用作名称空间(namespace)。这意味着你把数据保存在变量中,对他们按名称空间进行分组,使得他们处于同样的关系空间中——所谓的关系是使用标准 Python 句点属性标识。例如,你有一个本身没有任何属性的类,使用它仅对数据提供一个名字空间,让你的类拥有像 C 语言中的结构体(structure)一样的特性,或者换句话说,这样的类仅作为容器对象来共享名字空间。

示例如下:

class MyData(object):
  pass

mathObj = MyData()
mathObj.x = 4
mathObj.y = 5
mathObj.x + mathObj.y
9
mathObj.x \\* mathObj.y
20

方法
在 Python 中,方法定义在类定义中,但只能被实例所调用。也就是说,调用一个方法的最终途径必须是这样的:(1)定义类(和方法);(2)创建一个实例;(3)最后一步,用这个实例调用方法。例如:

class MyDataWithMethod(object): # 定义类
  def printFoo(self): # 定义方法
    print 'You invoked printFoo()!'

这里的 self 参数,它在所有的方法声明中都存在。这个参数代表实例对象本身,当你用实例调用方法时,由解释器传递给方法的,所以,你不需要自己传递 self 进来,因为它是自动传入的。

举例说明一下,假如你有一个带两参数的方法,所有你的调用只需要传递第二个参数。

下面是实例化这个类,并调用那个方法:

> > > myObj = MyDataWithMethod()
> > > myObj.printFoo()
You invoked printFoo()!

\\_init\\(),是一个特殊的方法。在 Python 中, \\init\\() 实际上不是一个构造器。你没有调用“new”来创建一个新对象。(Python 根本就没有“new”这个关键字)。取而代之, Python 创建实例后,在实例化过程中,调用 \\init\\_()方法,当一个类被实例化时,就可以定义额外的行为,比如,设定初始值或者运行一些初步诊断代码——主要是在实例被创建后,实例化调用返回这个实例之前,去执行某些特定的任务或设置。

创建一个类(类定义)

class AddrBookEntry(object):
  '''address book entry class'''
  def __init__(self, nm, ph): # 定义构造器
    self.name = nm       # 设置 name
    self.phone = ph       # 设置 phone
    print 'Created instance for:', self.name

  def updatePhone(self, newph):  # 定义方法
    self.phone = newph
    print 'Updated phone# for: ', self.name

在 AddrBookEntry 类的定义中,定义了两个方法: \\_init\\()和updatePhone()。\\init\\()在实例化时被调用,即,在AddrBookEntry()被调用时。你可以认为实例化是对 \\init\\()的一种隐式的调用,因为传给AddrBookEntry()的参数完全与\\init\\_()接收到的参数是一样的(除了self,它是自动传递的)。

创建实例(实例化)

> > > john = AddrBookEntry('John Doe', '408-555-1212') # 为 John Doe 创建实例
> > > jane = AddrBookEntry('Jane Doe', '650-555-1212') # 为 Jane Doe 创建实例

这就是实例化调用,它会自动调用 \\_init\\()。 self 把实例对象自动传入\\init\\_()。

另外,如果不存在默认的参数,那么传给 \\_init\\_() 的两个参数在实例化时是必须的。

访问实例属性

> > > john
> > > john.name
> > > jane.name
> > > jane.phone

一旦实例被创建后,就可以证实一下,在实例化过程中,我们的实例属性是否确实被 \\_init\\_() 设置了。我们可以通过解释器“转储”实例来查看它是什么类型的对象。

方法调用(通过实例)

> > > john.updatePhone('415-555-1212')  # 更新 John Doe 的电话
> > > john.phone

updatePhone()方法需要一个参数(不计 self 在内):新的电话号码。在 updatePhone()之后,立即检查实例属性,可以证实已生效。

方法与属性的小结
直接上代码,已经在里面有注释了

#coding:utf8

name = 'yangyanxing'
class Test():
  class kevin():
    var1 = '我是内部类'

  name = 'kvein'
  gae = '26'

  def fun1(self):
    print self.name
    print '我是公共方法'
    self.__fun2() #可以通过公有就去来调用私有方法,在调用的过程中可以进行更改

  def __fun2(self):
    print '我是私有方法'

  @classmethod
  def fun3(self): #可以不通过实例来访问这个类方法
    print '#'*40
    print self.name
    print '我是类方法'

  @staticmethod #静态方法,也是可以不通过实例对象就可以访问的方法但是在定义的时候不用加self
  def fun4():
    print Test.name
    print name #这里的name是全局变量
    Test.fun3()
    print '我是静态方法'

print Test.name #公有属性可以直接方法,不用实例化对象
yang = Test() #实例化一个类
interyang = Test.kevin() #实例化一个内部类
yang.fun1() #方法类里面的公共属性
print interyang.var1 # 访问内部类里的属性
Test.fun3()#访问类方法
Test.fun4()
#coding:utf8

class Test():
  var1 = '类的公有属性'
  __var2 = '类的私有属性'

  def fun(self):
    self.var2 = '对象的公有属性' # 这里定义了一个对象的公有属性
    self.__var3 = '对象的私有属性'# 这里定义了一个对象的私有属性
    var4 = '函数的局部变量' #这里定义了一个函数的局部变量,这里面的var4只有在函数内部使用

kevin = Test() #实例了一个对象
yang = Test() #又实例了另外一个对象
print kevin.var1
##print kevin.__var2 #这里将无法访问
kevin.fun()
print kevin.var2 #在没有调用fun函数之前是没有var2的
##print kevin.__var3 对象的私有属性是无法调用的
##print yang.var2 #这里因为没有调用yang的fun方法,所以还是无法访问yang里的var2

创建子类
靠继承来进行子类化是创建和定制新类型的一种方式,新的类将保持已存在类所有的特性,而不会改动原来类的定义。对于新类类型而言,这个新的子类可以定制只属于它的特定功能。除了与父类或基类的关系外,子类与通常的类没有什么区别,也像一般类一样进行实例化。注意下面,子类声明中提到了父类:

class EmplAddrBookEntry(AddrBookEntry):
  '''Employee Address Book Entry class''' # 员工地址簿类
  def __init__(self, nm, ph, id, em):
    AddrBookEntry.__init__(self, nm, ph)
    self.empid = id
    self.email = em

  def updateEmail(self, newem):
    self.email = newem
    print 'Updated e-mail address for:', self.name

现在我们创建了第一个子类, EmplAddrBookEntry。 Python 中,当一个类被派生出来,子类就继承了基类的属性,所以,在上面的类中,我们不仅定义了 \\_init\\_(),UpdateEmail()方法,而且 EmplAddrBookEntry 还从 AddrBookEntry 中继承了 updatePhone()方法。

如果需要,每个子类最好定义它自己的构造器,不然,基类的构造器会被调用。然而,如果子类重写基类的构造器,基类的构造器就不会被自动调用了——这样,基类的构造器就必须显式写出才会被执行,就像我们上面那样,用AddrBookEntry.\\_init\\_()设置名字和电话号码。我们的子类在构造器后面几行还设置了另外两个实例属性:员工ID和电子邮件地址。

注意,这里我们要显式传递 self 实例对象给基类构造器,因为我们不是在该实例中而是在一个子类实例中调用那个方法。因为我们不是通过实例来调用它,这种未绑定的方法调用需要传递一个适当的实例(self)给方法。

使用子类

> > > john = EmplAddrBookEntry('John Doe', '408-555-1212', 42, 'john@spam.doe')
> > > john
> > > john.name
> > > john.phone
> > > john.email
> > > john.updatePhone('415-555-1212')
> > > john.phone
> > > john.updateEmail('john@doe.spam')
> > > john.email
Python 相关文章推荐
利用python解决mysql视图导入导出依赖的问题
Dec 17 Python
完美解决安装完tensorflow后pip无法使用的问题
Jun 11 Python
详解django中使用定时任务的方法
Sep 27 Python
python实现统计文本中单词出现的频率详解
May 20 Python
基于django传递数据到后端的例子
Aug 16 Python
Django--权限Permissions的例子
Aug 28 Python
在pytorch 中计算精度、回归率、F1 score等指标的实例
Jan 18 Python
python 回溯法模板详解
Feb 26 Python
Pycharm中切换pytorch的环境和配置的教程详解
Mar 13 Python
pycharm中导入模块错误时提示Try to run this command from the system terminal
Mar 26 Python
Keras—embedding嵌入层的用法详解
Jun 10 Python
Visual Studio Code搭建django项目的方法步骤
Sep 17 Python
Python使用cookielib模块操作cookie的实例教程
Jul 12 #Python
Python网络编程中urllib2模块的用法总结
Jul 12 #Python
Python中内置的日志模块logging用法详解
Jul 12 #Python
Swift 3.0在集合类数据结构上的一些新变化总结
Jul 11 #Python
浅析Python的web.py框架中url的设定方法
Jul 11 #Python
深入解析Python的Tornado框架中内置的模板引擎
Jul 11 #Python
使用Python的Tornado框架实现一个Web端图书展示页面
Jul 11 #Python
You might like
这东西价格,可以买几台TECSUN S-2000
2021/03/02 无线电
分享50个提高PHP执行效率的技巧
2015/12/26 PHP
PHP生成推广海报的方法分享
2018/04/22 PHP
基于jquery循环map功能的代码
2011/02/26 Javascript
纯js网页画板(Graphics)类简介及实现代码
2012/12/24 Javascript
js Map List 遍历使用示例
2013/07/10 Javascript
javascript 实现 原路返回
2015/01/21 Javascript
javascript实现简单的省市区三级联动
2015/05/14 Javascript
javascript实现的闭包简单实例
2015/07/17 Javascript
AngularJS数据源的多种获取方式汇总
2016/02/02 Javascript
Vue.js第一天学习笔记(数据的双向绑定、常用指令)
2016/12/01 Javascript
JavaScript无缝滚动效果的实例代码
2017/03/27 Javascript
微信小程序商城项目之侧栏分类效果(1)
2017/04/17 Javascript
jQuery Autocomplete简介_动力节点Java学院整理
2017/07/17 jQuery
基于Node.js模板引擎教程-jade速学与实战1
2017/09/17 Javascript
js截取字符串功能的实现方法
2017/09/27 Javascript
Vue项目数据动态过滤实践及实现思路
2018/09/11 Javascript
MockJs结合json-server模拟后台数据
2020/08/26 Javascript
[01:00:06]加油DOTA_EP01_网络版
2014/08/09 DOTA
[49:35]LGD vs OG 2018国际邀请赛淘汰赛BO3 第二场 8.25
2018/08/29 DOTA
Python内置函数Type()函数一个有趣的用法
2015/02/18 Python
Python 实现一个颜色色值转换的小工具
2016/12/06 Python
win10下Python3.6安装、配置以及pip安装包教程
2017/10/01 Python
浅谈Python中函数的定义及其调用方法
2019/07/19 Python
Python 多线程,threading模块,创建子线程的两种方式示例
2019/09/29 Python
10个python3常用排序算法详细说明与实例(快速排序,冒泡排序,桶排序,基数排序,堆排序,希尔排序,归并排序,计数排序)
2020/03/17 Python
利用Python实现Excel的文件间的数据匹配功能
2020/06/16 Python
Python代码需要缩进吗
2020/07/01 Python
Pytorch - TORCH.NN.INIT 参数初始化的操作
2021/02/27 Python
Etam俄罗斯:法国女士内衣和家居服网上商店
2019/10/30 全球购物
Bloomingdale’s阿联酋:选购奢华时尚、美容及更多
2020/09/22 全球购物
招商专员岗位职责
2014/02/08 职场文书
个人函授自我鉴定
2014/03/25 职场文书
带病坚持工作事迹
2014/05/03 职场文书
贷款委托书
2014/08/01 职场文书
继续教育心得体会(共6篇)
2016/01/19 职场文书