Python中的对象,方法,类,实例,函数用法分析


Posted in Python onJanuary 15, 2015

本文实例分析了Python中的对象,方法,类,实例,函数用法。分享给大家供大家参考。具体分析如下:

Python是一个完全面向对象的语言。不仅实例是对象,类,函数,方法也都是对象。

class Foo(object):

    static_attr = True

    def method(self):

        pass

foo = Foo()

这段代码实际上创造了两个对象,Foo和foo。而Foo同时又是一个类,foo是这个类的实例。
在C++里类型定义是在编译时完成的,被储存在静态内存里,不能轻易修改。在Python里类型本身是对象,和实例对象一样储存在堆中,对于解释器来说类对象和实例对象没有根本上的区别。
在Python中每一个对象都有自己的命名空间。空间内的变量被存储在对象的__dict__里。这样,Foo类有一个__dict__, foo实例也有一个__dict__,但这是两个不同的命名空间。
所谓“定义一个类”,实际上就是先生成一个类对象,然后执行一段代码,但把执行这段代码时的本地命名空间设置成类的__dict__. 所以你可以写这样的代码:
>>> class Foo(object):

...     bar = 1 + 1

...     qux = bar + 1

...     print "bar: ", bar

...     print "qux: ", qux

...     print locals()

...

bar:  2

qux:  3

{'qux': 3, '__module__': '__main__', 'bar': 2}

>>> print Foo.bar, Foo.__dict__['bar']

2 2

>>> print Foo.qux, Foo.__dict__['qux']

3 3

所谓“定义一个函数”,实际上也就是生成一个函数对象。而“定义一个方法”就是生成一
个函数对象,并把这个对象放在一个类的__dict__中。下面两种定义方法的形式是等价的:

>>> class Foo(object):

...     def bar(self):

...         return 2

...

>>> def qux(self):

...     return 3

...

>>> Foo.qux = qux

>>> print Foo.bar, Foo.__dict__['bar']
>>> print Foo.qux, Foo.__dict__['qux']
>>> foo = Foo()

>>> foo.bar()

2

>>> foo.qux()

3

而类继承就是简单地定义两个类对象,各自有不同的__dict__:

>>> class Cheese(object):

...     smell = 'good'

...     taste = 'good'

...

>>> class Stilton(Cheese):

...     smell = 'bad'

...

>>> print Cheese.smell

good

>>> print Cheese.taste

good

>>> print Stilton.smell

bad

>>> print Stilton.taste

good

>>> print 'taste' in Cheese.__dict__

True

>>> print 'taste' in Stilton.__dict__

False

复杂的地方在`.`这个运算符上。对于类来说,Stilton.taste的意思是“在Stilton.__dict__中找'taste'. 如果没找到,到父类Cheese的__dict__里去找,然后到父类的父类,等等。如果一直到object仍没找到,那么扔一个AttributeError.”
实例同样有自己的__dict__:

>>> class Cheese(object):

...     smell = 'good'

...     taste = 'good'

...     def __init__(self, weight):

...         self.weight = weight

...     def get_weight(self):

...         return self.weight

...

>>> class Stilton(Cheese):

...     smell = 'bad'

...

>>> stilton = Stilton('100g')

>>> print 'weight' in Cheese.__dict__

False

>>> print 'weight' in Stilton.__dict__

False

>>> print 'weight' in stilton.__dict__

True

不管__init__()是在哪儿定义的, stilton.__dict__与类的__dict__都无关。
Cheese.weight和Stilton.weight都会出错,因为这两个都碰不到实例的命名空间。而
stilton.weight的查找顺序是stilton.__dict__ => Stilton.__dict__ =>
Cheese.__dict__ => object.__dict__. 这与Stilton.taste的查找顺序非常相似,仅仅是
在最前面多出了一步。

方法稍微复杂些。

>>> print Cheese.__dict__['get_weight']
>>> print Cheese.get_weight
>>> print stilton.get_weight

<__main__.Stilton object at 0x7ff820669190>>

我们可以看到点运算符把function变成了unbound method. 直接调用类命名空间的函数和点
运算返回的未绑定方法会得到不同的错误:
>>> Cheese.__dict__['get_weight']()

Traceback (most recent call last):

  File "", line 1, in

TypeError: get_weight() takes exactly 1 argument (0 given)

>>> Cheese.get_weight()

Traceback (most recent call last):

  File "", line 1, in

TypeError: unbound method get_weight() must be called with Cheese instance as

first argument (got nothing instead)

但这两个错误说的是一回事,实例方法需要一个实例。所谓“绑定方法”就是简单地在调用方法时把一个实例对象作为第一个参数。下面这些调用方法是等价的:
>>> Cheese.__dict__['get_weight'](stilton)

'100g'

>>> Cheese.get_weight(stilton)

'100g'

>>> Stilton.get_weight(stilton)

'100g'

>>> stilton.get_weight()

'100g'

最后一种也就是平常用的调用方式,stilton.get_weight(),是点运算符的另一种功能,将stilton.get_weight()翻译成stilton.get_weight(stilton).
这样,方法调用实际上有两个步骤。首先用属性查找的规则找到get_weight, 然后将这个属性作为函数调用,并把实例对象作为第一参数。这两个步骤间没有联系。比如说你可以这样试:
>>> stilton.weight()

Traceback (most recent call last):

  File "", line 1, in

TypeError: 'str' object is not callable

先查找weight这个属性,然后将weight做为函数调用。但weight是字符串,所以出错。要注意在这里属性查找是从实例开始的:
>>> stilton.get_weight = lambda : '200g'

>>> stilton.get_weight()

'200g'

但是
>>> Stilton.get_weight(stilton)

'100g'

Stilton.get_weight的查找跳过了实例对象stilton,所以查找到的是没有被覆盖的,在Cheese中定义的方法。

getattr(stilton, 'weight')和stilton.weight是等价的。类对象和实例对象没有本质区别,getattr(Cheese, 'smell')和Cheese.smell同样是等价的。getattr()与点运算符相比,好处是属性名用字符串指定,可以在运行时改变。

__getattribute__()是最底层的代码。如果你不重新定义这个方法,object.__getattribute__()和type.__getattribute__()就是getattr()的具体实现,前者用于实例,后者用以类。换句话说,stilton.weight就是object.__getattribute__(stilton, 'weight'). 覆盖这个方法是很容易出错的。比如说点运算符会导致无限递归:

def __getattribute__(self, name):

        return self.__dict__[name]

__getattribute__()中还有其它的细节,比如说descriptor protocol的实现,如果重写很容易搞错。

__getattr__()是在__dict__查找没找到的情况下调用的方法。一般来说动态生成属性要用这个,因为__getattr__()不会干涉到其它地方定义的放到__dict__里的属性。

>>> class Cheese(object):

...     smell = 'good'

...     taste = 'good'

...

>>> class Stilton(Cheese):

...     smell = 'bad'

...     def __getattr__(self, name):

...         return 'Dynamically created attribute "%s"' % name

...

>>> stilton = Stilton()

>>> print stilton.taste

good

>>> print stilton.weight

Dynamically created attribute "weight"

>>> print 'weight' in stilton.__dict__

False

由于方法只不过是可以作为函数调用的属性,__getattr__()也可以用来动态生成方法,但同样要注意无限递归:
>>> class Cheese(object):

...     smell = 'good'

...     taste = 'good'

...     def __init__(self, weight):

...         self.weight = weight

...

>>> class Stilton(Cheese):

...     smell = 'bad'

...     def __getattr__(self, name):

...         if name.startswith('get_'):

...             def func():

...                 return getattr(self, name[4:])

...             return func

...         else:

...             if hasattr(self, name):

...                 return getattr(self, name)

...             else:

...                 raise AttributeError(name)

...

>>> stilton = Stilton('100g')

>>> print stilton.weight

100g

>>> print stilton.get_weight
>>> print stilton.get_weight()

100g

>>> print stilton.age

Traceback (most recent call last):

  File "", line 1, in

  File "", line 12, in __getattr__

AttributeError: age

希望本文所述对大家的Python程序设计有所帮助。

Python 相关文章推荐
Python 列表list使用介绍
Nov 30 Python
Python闭包实现计数器的方法
May 05 Python
在Python的Django框架中更新数据库数据的方法
Jul 17 Python
python3操作mysql数据库的方法
Jun 23 Python
Python3利用Dlib19.7实现摄像头人脸识别的方法
May 11 Python
对pandas将dataframe中某列按照条件赋值的实例讲解
Nov 29 Python
详解python 3.6 安装json 模块(simplejson)
Apr 02 Python
Django 自定义分页器的实现代码
Nov 24 Python
2020新版本pycharm+anaconda+opencv+pyqt环境配置学习笔记,亲测可用
Mar 24 Python
python实现对变位词的判断方法
Apr 05 Python
如何在python中执行另一个py文件
Apr 30 Python
TensorFlow低版本代码自动升级为1.0版本
Feb 20 Python
Python转换HTML到Text纯文本的方法
Jan 15 #Python
python中os操作文件及文件路径实例汇总
Jan 15 #Python
python私有属性和方法实例分析
Jan 15 #Python
python实现堆栈与队列的方法
Jan 15 #Python
python多线程用法实例详解
Jan 15 #Python
Python中os.path用法分析
Jan 15 #Python
python静态方法实例
Jan 14 #Python
You might like
COM in PHP (winows only)
2006/10/09 PHP
php smarty模版引擎中变量操作符及使用方法
2009/12/11 PHP
php中修改浏览器的User-Agent来伪装你的浏览器和操作系统
2011/07/29 PHP
PHP设计模式之观察者模式定义与用法示例
2018/08/04 PHP
PHP图像处理技术实例总结【绘图、水印、验证码、图像压缩】
2018/12/08 PHP
PHP切割汉字的常用方法实例总结
2019/04/27 PHP
总结一些js自定义的函数
2006/08/05 Javascript
JavaScript解析URL参数示例代码
2013/08/12 Javascript
Node.js实现在目录中查找某个字符串及所在文件
2014/09/03 Javascript
JS获取iframe中longdesc属性的方法
2015/04/01 Javascript
微信小程序左右滑动切换页面详解及实例代码
2017/02/28 Javascript
Vue js 的生命周期(看了就懂)(推荐)
2019/03/29 Javascript
改变layer confirm弹窗按钮的颜色方法
2019/09/12 Javascript
js实现表单项的全选、反选及删除操作示例
2020/06/05 Javascript
Vue环境搭建+VSCode+Win10的详细教程
2020/08/19 Javascript
[03:49]辉夜杯现场龙骑士COSER秀情商“我喜欢芬队!”
2015/12/27 DOTA
[54:18]DOTA2-DPC中国联赛 正赛 PSG.LGD vs LBZS BO3 第一场 1月22日
2021/03/11 DOTA
详解Python logging调用Logger.info方法的处理过程
2019/02/12 Python
python爬取酷狗音乐排行榜
2019/02/20 Python
win8.1安装Python 2.7版环境图文详解
2019/07/01 Python
Python Django 实现简单注册功能过程详解
2019/07/29 Python
python是否适合网页编程详解
2019/10/04 Python
keras 获取某层的输入/输出 tensor 尺寸操作
2020/06/10 Python
利用Vscode进行Python开发环境配置的步骤
2020/06/22 Python
基于python实现ROC曲线绘制广场解析
2020/06/28 Python
使用CSS3制作饼状旋转载入效果的实例
2015/06/23 HTML / CSS
美国当红的名品折扣网:Gilt Groupe
2016/08/15 全球购物
美国最流行的男士时尚网站:Touch of Modern
2018/02/05 全球购物
输入N,打印N*N矩阵
2012/02/20 面试题
售后主管岗位职责
2013/12/08 职场文书
大学生职业生涯规划范文——找准自我,定位人生
2014/01/23 职场文书
2014年社区植树节活动方案
2014/02/28 职场文书
观看《信仰》心得体会
2016/01/15 职场文书
python生成随机数、随机字符、随机字符串
2021/04/06 Python
教你使用vscode 搭建react-native开发环境
2021/07/07 Javascript
C3 线性化算法与 MRO之Python中的多继承
2021/10/05 Python