Python实例一个类背后发生了什么


Posted in Python onFebruary 09, 2016

首先来看一个例子,正常情况下我们定义并且实例一个类如下

class Foo(object):
 
  def __init__(self):
    pass
 
obj = Foo()  # obj是通过Foo类实例化的对象

上述代码中,obj 是通过 Foo 类实例化的对象,其实,不仅 obj 是一个对象,Foo类本身也是一个对象,因为在Python中一切事物都是对象。

print type(obj) # 输出: Foo 表示,obj 对象由Foo类创建
print type(Foo) # 输出:type表示,Foo类对象由 type 类创建
如果按照一切事物都是对象的理论:对象是通过执行Foo类的构造方法创建,那么Foo类对象应该也是通过执行某个类的 构造方法 创建。

一、两个基本的类
这里和有必要提到一下在Python中有两个最基本的对象,<type ‘object'> 和 <type ‘type'>,这两个对象是所有对象的起源。

1、<type ‘type'>的类型是<type ‘type'>(它本身),<type ‘type'>的父类是<type ‘object'>
2、<type ‘object'>的类型是<type ‘type'>,<type ‘object'>没有父类
在Python对象系统中,<type ‘object'>和<type ‘type'>的关系就像鸡和蛋的关系,不能说谁先于(创建)谁,两者是相互依赖的,共同构成了Python对象系统的基础.有点难理解,不过没有关系,知道有种这么两个东西就好了,比较我们也不是设计Python的人,没有必要搞的那么清楚.

二、类的创建
主要有两种方式,不过本质上都是一样的,都是通过type类来实例一个用户类

//普通方式
  class Foo(object):
   
  def func(self):
  print 'hello wupeiqi'

//特殊方式(type类的构造函数)
  def func(self):
    print 'hello wupeiqi'
  
  Foo = type('Foo',(object,), {'func': func})
 #type第一个参数:类名
 #type第三个参数:类的成员

由上面可以看出来 , Foo类是由type类实例而来,那么具体的创建的过程是怎么样的呢,接着往下面看:

来了解几个概念
new __init()__metaclass__:

  • __new__函数是实例一个类所要调用的函数,每当我们调用obj = Foo()来实例一个类时,都是先调用__new__()
  • 然后再调用__init__()函数初始化实例. __init__()在__new__()执行后执行,
  • 类中还有一个属性 __metaclass__,其用来表示该类由 谁 来实例化创建,所以,我们可以为 __metaclass__ 设置一个type类的派生类,从而查看 类 创建的过程。

三、阐述运行过程

1、mytype产生一个叫做Foo的实例,主要的原理就是设置了,__metaclass__=MyTypoe,这样就指定mytype类来实例foo类,如果Python没有找到__metaclass__,它会继续在(父类)中寻找 __metaclass__属性,并尝试做和前面同样的操作。如果Python在任何父类中都找不到__metaclass__,它就会在模块层次中去寻找__metaclass__,并尝试做同样的操作。如果还是找不到__metaclass__,Python就会用内置的type来创建这个类对象。
2、mytype类中的__new__方法返回了一个对象,所有的Python实例都是这句代码创建的type.__new__(cls,name,bases,attrs)
3、mytype的__init__()函数初始化Foo类,在这里我们可以和在__new__()函数一样设置Foo类的attr属性,比如类中的方法,字段属性等
4、和Foo类的创建过程一样,studen类继承了Foo类,所以重复123步骤,得到一个studen类
5、当用户使用Foo()或者studen()来实例类时,会默认调用类中的_new_()方法,要是之类里面没有这个方法就到父类里面寻找__new__(),我们可以充分利用这个new函数,比如来实现Python中的单例模式,或者对类成员进行批量的修改等等.
6、产生了一个实例后马上执行__init__()函数,进行初始化实例,
7、由上面的运行结果可以看出,其中Foo和studen类的类型是<class '__main__.MyType'>,这也证明了Foo类和studen类都是由MyType实例而来…而第三个例子,可以看出Foo2类的类型是<type 'type'>,这个并不例外,Foo2继承了object类,(这个,还是跳过吧,前面已经讲过type和object相爱相杀的关系了)
总结
首先,你知道了类其实是能够创建出类实例的对象。好吧,事实上,类本身也是实例,当然,它们是元类的实例。Python中的一切都是对象,它们要么是类的实例,要么是元类的实例,除了type。type实际上是它自己的元类,在纯Python环境中这可不是你能够做到的,这是通过在实现层面耍一些小手段做到的。其次,元类是很复杂的。对于非常简单的类,你可能不希望通过使用元类来对类做修改。你可以通过其他两种技术来修改类:

1、Monkey patching
2、class decora

以上就是本文的详细内容,希望对大家的学习有所帮助。

Python 相关文章推荐
python实现定时同步本机与北京时间的方法
Mar 24 Python
Python的Django框架中设置日期和字段可选的方法
Jul 17 Python
python爬虫获取京东手机图片的图文教程
Dec 29 Python
python3 flask实现文件上传功能
Mar 20 Python
pandas 转换成行列表进行读取与Nan处理的方法
Oct 30 Python
python写程序统计词频的方法
Jul 29 Python
Python 从subprocess运行的子进程中实时获取输出的例子
Aug 14 Python
python tkinter canvas使用实例
Nov 04 Python
tensorflow多维张量计算实例
Feb 11 Python
python输入中文的实例方法
Sep 14 Python
scrapy利用selenium爬取豆瓣阅读的全步骤
Sep 20 Python
Python3.9新特性详解
Oct 10 Python
Python中的条件判断语句基础学习教程
Feb 07 #Python
Python模拟登录验证码(代码简单)
Feb 06 #Python
Python上传package到Pypi(代码简单)
Feb 06 #Python
深入讲解Java编程中类的生命周期
Feb 05 #Python
python&amp;MongoDB爬取图书馆借阅记录
Feb 05 #Python
Python中常用操作字符串的函数与方法总结
Feb 04 #Python
Python中的字符串类型基本知识学习教程
Feb 04 #Python
You might like
php统计文件大小,以GB、MB、KB、B输出
2011/05/29 PHP
Android ProgressBar进度条和ProgressDialog进度框的展示DEMO
2013/06/19 PHP
php 启动时报错的简单解决方法
2014/01/27 PHP
JSON两种结构之对象和数组的理解
2016/07/19 PHP
深入理解PHP的远程多会话调试
2017/09/21 PHP
php生成静态页面并实现预览功能
2019/06/27 PHP
jQuery中RadioButtonList的功能及用法实例介绍
2013/08/23 Javascript
Js参数值中含有单引号或双引号问题的解决方法
2013/11/06 Javascript
React-intl 实现多语言的示例代码
2017/11/03 Javascript
浅谈es6中export和export default的作用及区别
2018/02/07 Javascript
详解关于Vue版本不匹配问题(Vue packages version mismatch)
2018/09/17 Javascript
Vue中使用create-keyframe-animation与动画钩子完成复杂动画
2019/04/09 Javascript
javascript数组常见操作方法实例总结【连接、添加、删除、去重、排序等】
2019/06/13 Javascript
jQuery实现移动端图片上传预览组件的方法分析
2020/05/01 jQuery
Python中遇到的小问题及解决方法汇总
2017/01/11 Python
Python常用算法学习基础教程
2017/04/13 Python
python实现整数的二进制循环移位
2019/03/08 Python
Python中__repr__和__str__区别详解
2019/11/07 Python
解决django model修改添加字段报错的问题
2019/11/18 Python
tensorflow查看ckpt各节点名称实例
2020/01/21 Python
Python range与enumerate函数区别解析
2020/02/28 Python
python GUI库图形界面开发之PyQt5布局控件QGridLayout详细使用方法与实例
2020/03/06 Python
Django用户身份验证完成示例代码
2020/04/03 Python
pytorch __init__、forward与__call__的用法小结
2021/02/27 Python
html5跨域通讯之postMessage的用法总结
2013/11/07 HTML / CSS
玖熙女鞋美国官网:Nine West
2016/10/06 全球购物
芬兰汽车配件商店:Autonvaraosat24
2017/01/30 全球购物
七年级生物教学反思
2014/01/30 职场文书
优秀信贷员先进事迹
2014/01/31 职场文书
《玩具柜台前的孩子》教学反思
2014/02/13 职场文书
激励口号大全
2014/06/17 职场文书
4s店销售经理岗位职责
2014/07/19 职场文书
以幸福为主题的活动方案
2014/08/22 职场文书
2014国庆节幼儿园亲子活动方案
2014/09/16 职场文书
防止web项目中的SQL注入
2021/12/06 MySQL
Python+Matplotlib图像上指定坐标的位置添加文本标签与注释
2022/04/11 Python