Python中类调用__new__()类方法来创建实例,调用__init__()方法来初始化对象,对象的销毁则调用__del__()方法。

__new__()方法第一个参数为类cls,通常返回cls的一个实例,然后新实例的__init__()方法将以类似于__init__(self[, ...])的方式被调用,self是创建的新的实例,其它的参数和传递给__new__()的参数一样。

如果__new__()方法不返回cls 的实例,那么__init__()将不会被调用。

对于一个类来说:

  1. class Foo(object):
  2. def __init__(self, name):
  3. self.name = name
  4.  
  5. f = Foo('bar')

当进行f = Foo('bar')的操作时,实际上执行以下步骤来创建和初始化实例:

  1. f = Foo.__new__(Foo, 'bar')
  2. if isinstance(f, Foo):
  3. Foo.__init__(f, 'bar')

__new__(cls[, ...])

__new__()方法是一个静态类方法,以实例的类为第一个参数。其余的参数是传递给对象构造表达式的参数,与传递给__init__()的相同。

一般通过使用super(currentclass, cls).__new__(cls[, ...])来调用超类的__new__()方法创建类的一个新的实例。

用户自定义对象的时候,很少定义__new__()方法。如果定义的话,一般是在定义元类或者定义一些继承自不可变的内置数据类型(整数、字符串、元组等)的对象的时候。

定义一些不可变数据类型的时候,因为对象是不可变的,那么在__init__()中是无法对对象进行修改的,而__new__()方法是在实例创建之前执行的:

  1. class Upperstr(str):
  2. def __new__(cls, value=''):
  3. return str.__new__(cls, value.upper())

上述Upperstr类创造的字符串实例全是大写的:

  1. >>> u = Upperstr('hello world')
  2. >>> u
  3. 'HELLO WORLD'

也可以在单例模式中使用__new__()方法创建唯一的实例,放在类变量中,以后每次创建都返回这一个实例:

  1. class Singleton(object):
  2. def __new__(cls, *args, **kwargs):
  3. if not hasattr(cls, '_inst'):
  4. cls._inst=super(Singleton, cls).__new__(cls, *args, **kwargs)
  5. return cls._inst

__init__(self[, ...])

__init__()方法用来初始化对象的属性,在实例创建完成时调用。参数为传递给类构造器表达式的那些参数。如果基类具有__init__()方法,那么继承的类的__init__()方法,而且必须显式调用:BaseClass.__init__(self, [args...])。

构造器的一个特殊限制是不可以返回任何值,这样做将导致运行时抛出一个TypeError。

__init__()方法一般含有大量的self.attr = attr的赋值语句:

  1. class Foo(object):
  2. def __init__(self, name, color, num=1):
  3. self.name = name
  4. self.color = color
  5. self.num = num
  6.  
  7. f = Foo('bar', 'yellow')

如果不想重复这样的赋值操作,有以下几个办法:

  1. class Foo(object):
  2. def __init__(self, name, color, num=1):
  3. print locals()
  4.  
  5. f = Foo('bar', 'yellow')

__init__()方法的局部命名空间如下:

  1. {'color': 'yellow', 'self': <__main__.Foo object at 0x00000000055C4780>, 'num': 1, 'name': 'bar'}

self变量为创建的实例本身,其余要给实例赋值的变量name,color,num均在其中。而实例属性是存储在实例的__dict__变量中,我们只需要将这些要赋值的变量放入__dict__即可:

  1. class Foo(object):
  2. def __init__(self, name, color, num=1):
  3. self.__dict__.update(locals())
  4. del self.__dict__['self']
  1. >>> f = Foo('bar', 'yellow')
  2. >>> f.name
  3. 'bar'
  4. >>> f.color
  5. 'yellow'
  6. >>> f.num
  7. 1

这样当实例属性很多的时候,就简洁了很多。但当使用特性或者描述符来包装对象的属性时,会产生问题,因为特性或者描述符包装的属性并不保存于__dict__之中,解决方案是使用setattr内置函数:

  1. def attr_from_locals(locals_dict):
  2. self = locals_dict.pop('self')
  3. for k,v in locals_dict.items():
  4. setattr(self, k, v)
  5.  
  6. class Foo(object):
  7. def __init__(self, name, color, num=1):
  8. attr_from_locals(locals())

如果__init__()方法中使用了一些不用赋值给实例的本地变量的时候,再使用locals()会包含这些本地变量,我们可以再精准的传递__init__()中的参数。

通过一个函数f的f.__code__属性可以得到函数的字节码对象c,字节码对象的属性c.co_varnames可以得到局部变量名称的元组(函数参数在前,函数内局部变量在后),c.co_argcount可以得到函数位置参数的数量,利用切片可以去除函数内局部变量:

  1. def attr_from_locals(locals_dict):
  2. self = locals_dict.pop('self')
  3. code = self.__init__.__func__.__code__
  4. args = code.co_varnames[1:code.co_argcount]
  5. for k in args:
  6. setattr(self, k, locals_dict[k])
  7.  
  8. class Foo(object):
  9. def __init__(self, name, color, num=1):
  10. x = 1
  11. attr_from_locals(locals())
  12.  
  13. f = Foo('bar', 'yellow')

可以看到x变量并未赋值给实例:

  1. >>> f.name
  2. 'bar'
  3. >>> f.color
  4. 'yellow'
  5. >>> f.num
  6. 1
  7. >>> f.x
  8.  
  9. Traceback (most recent call last):
  10. File "<pyshell#87>", line 1, in <module>
  11. f.x
  12. AttributeError: 'Foo' object has no attribute 'x'

__del__(self)

当实例要被销毁的时候,会调用对象的__del__()方法。注意当使用del a这样的语句时,并不会调用__del__()方法,只是减少一个引用计数,当对象的引用计数减至0时,才会调用__del__()。

__del__()方法也很少有必要定义,定义了此方法的实例,无法被python的循环垃圾收集器收集。

需要执行清除操作时(关闭文件、关闭网络连接等)可以定义,当然更好的办法是定义一个close()方法显式的执行关闭操作。

Python中的对象行为与特殊方法(一)对象的创建与销毁的更多相关文章

  1. Python中的str与unicode处理方法

    Python中的str与unicode处理方法 2015/03/25 · 基础知识 · 3 评论· Python 分享到:42 原文出处: liuaiqi627 的博客    python2.x中处理 ...

  2. python中的魔术属性与魔法方法

    1.魔法属性 · 1.1__doc__魔法属性  表示类的描述信息 class Fo: """ 这是今天第一个魔术属性__doc__""" ...

  3. 第7.22节 Python中使用super调用父类的方法

    第7.22节 Python中使用super调用父类的方法 前面章节很多地方都引入了super方法,这个方法就是访问超类这个类对象的.由于super方法的特殊性,本节单独谈一谈super方法. 一.su ...

  4. Python中异步协程的使用方法介绍

    1. 前言 在执行一些 IO 密集型任务的时候,程序常常会因为等待 IO 而阻塞.比如在网络爬虫中,如果我们使用 requests 库来进行请求的话,如果网站响应速度过慢,程序一直在等待网站响应,最后 ...

  5. python中常用的导包的方法和常用的库

    python中常用的导包的方法               导入包和包名的方法:1.import package.module 2.from package.module import  * 例一: ...

  6. 举例详解Python中的split()函数的使用方法

    这篇文章主要介绍了举例详解Python中的split()函数的使用方法,split()函数的使用是Python学习当中的基础知识,通常用于将字符串切片并转换为列表,需要的朋友可以参考下   函数:sp ...

  7. Python中os和shutil模块实用方法集…

    Python中os和shutil模块实用方法集锦 类型:转载 时间:2014-05-13 这篇文章主要介绍了Python中os和shutil模块实用方法集锦,需要的朋友可以参考下 复制代码代码如下: ...

  8. Python中os和shutil模块实用方法集锦

    Python中os和shutil模块实用方法集锦 类型:转载 时间:2014-05-13 这篇文章主要介绍了Python中os和shutil模块实用方法集锦,需要的朋友可以参考下 复制代码代码如下: ...

  9. python中执行shell的两种方法总结

    这篇文章主要介绍了python中执行shell的两种方法,有两种方法可以在Python中执行SHELL程序,方法一是使用Python的commands包,方法二则是使用subprocess包,这两个包 ...

  10. Python中常见字符串去除空格的方法总结

    Python中常见字符串去除空格的方法总结 1:strip()方法,去除字符串开头或者结尾的空格>>> a = " a b c ">>> a.s ...

随机推荐

  1. Cocos Creator实现的《点我+1》

    一.前言 在学习Cocos中,需要一些东西来练手,于是前段时间就开发仿照一款公司之前的产品<点我+1>来做,仿照过程中,所有的算法逻辑都是自己研究的,并没有参考公司代码,也没有使用公司的美 ...

  2. Cocos Creator 为Button添加事件的两种方法

    Button添加事件 Button 目前只支持 Click 事件,即当用户点击并释放 Button 时才会触发相应的回调函数.通过脚本代码添加回调方法一这种方法添加的事件回调和使用编辑器添加的事件回调 ...

  3. Spring+SpringMVC+MyBatis整合配置

    前端控制器 web.xml <?xml version="1.0" encoding="UTF-8"?> <web-app version=& ...

  4. git 语法

    $ git init  // 初始化一个Git仓库 会生成一个.git目录 $ git status   // 查看仓库的状态 $ git add .   // 将所有修改添加到暂存区 $git ad ...

  5. ssh整合not found class 异常总结

    (1)org.apache.tomcat.dbcp.dbcp.SQLNestedException: Cannot load JDBC driver class 'com.microsoft.sqls ...

  6. SQL Server如何将Id相同的字段合并,并且以逗号隔开

    需要用到stuff函数: 例: id             name           1               张三           1               李四       ...

  7. scala mysql jdbc oper

    package egsql import java.util.Properties import com.sun.org.apache.xalan.internal.xsltc.compiler.ut ...

  8. schame定义及用处

    一.schame详解 http://www.cnblogs.com/Neo-ds/p/4790413.html 1.先明确一点,SQL Server中模式(schema)这个概念是在2005的版本里才 ...

  9. C# vs2015单元测试测试资源管理器不显示测试方法

    问题描述:在用VS2015用测试框架NUnit单元测试的时候,测试资源管理器死活不出现测试方法,无法运行单元测试模块 现象如下图: 原因:nunit版本不对应 解决方案:下载nunit3.0及往上的版 ...

  10. html5-新增表单的小结details summary

    <!DOCTYPE html><html lang="en"><head>    <meta charset="UTF-8&qu ...