数据模型(译)

 
image.png

1 对象(object)、类型(type)和值(value)

python中所有的数据都是通过对象(object)或者对象之间的关系来表示

每个对象(object)都有ID、类型(type)和数值(value)

一旦对象创建,它的ID便固定不变,可以理解成对象存放在内存中的地址;is操作就是比较两个对象的ID,而id()函数则是返回对象ID的整数表达式

对象的类型(type)决定了对象的行为,以及决定对象可能的值(value);type()函数返回对象的类型(type本身也是对象);类似于ID,对象的类型一般情况下也是不可变的

部分对象的值(value)是可变的,我们称之为可变类型(mutable);部分对象的值是不可变的,我们称这为不可变类型(immutable);当一个不可变容器(如tuple)包含可变类型的对象引用,虽然引用对象的值可以改变,但我们依然认为此容器的值不是可变的,因为容器本身包含的对象不可变,所以严格来讲,不可变类型并不意味着数值(value)不可更改。对象是否可变由类型决定,比如数字(numbers)、字符串(strings)和元组(tuples)是不可变的;字典(dictionaries)和列表(lists)是可变的

对象从不显示地销毁,当没有引用指向这些对象时,它们就可能被回收(GC)

注意,try...except语句会使对象保持存活

有些对象指向外部资源,如打开的文件或窗口;当对象被回归时,资源也一并释放。然而由于回收机制无法确定触发,所以类似的对象提供了显示的方式来释放外部资源,通常是close()方法。程序建议使用try...finally或者with语句来显示关闭

2 特殊方法

类可以通过定义具有特殊名称的方法来实现由特殊语法调用的某些操作,这是python的运算符重载方法,允许类根据语言运算符定义自己的行为

例如,如果一个类定义了__getitem__()的方法,并且x是该类的一个实例,那么x[i]大致相当于type(x).__getitem()__(x, i)

将特殊方法设置为None,意味着相关操作不可用;比如,将类的__iter()__方法设置为None,类将无法迭代,因此调用iter()方法时会抛出类型错误(TypeError)的异常

2.1 基本定制

object.__new__(cls[, ...])

创建类的实例,__new__()是静态方法,它将请求实例的类作为第一个参数,其余参数传递给对象构造函数,__new__()的返回值应该是新的对象实例(通常是cls的实例)

典型的实现__new__(),是使用适当的参数调用超类的方法,然后在返回之前根据需要,修改新创建的类的实例,super().__new__(cls[, ...])

如果__new__()返回cls的实例,那么实例的__init__()方法将被调用,如__init__(self [, ...]),其中self 代表新实例,其余的参数与传入__new__()方法的参数相同

如果__new__()没有返回类的实例,那么__init__()方法将不会调用

__new__()意在允许不可变类型(如数字、字符 串、元组)的子类自定义实例的创建;另外,也可以创建自定义的元类(metaclass)来定制类的创建

object.__init__(self, [...])

在实例被创建(即__new__())后,尚未返回给调用者之前调用,参数与传递给类的构造器表达式一致。如果基类有__init__()方法,子类的__init__()方法,如果存在的话,必须显示调用来保证实例的基类实例化操作,如super().__init__([args...])

因为__new__()__init__()共同完成对象的创建(new用来创建,init用来初始化),所以__init__()不允许非空值返回,不然的话会在运行时抛出类型错误异常

object.__del__(self)

在实例将要销毁时调用

object.__repr__(self)

通过repr()内置函数调用,用来获取对象的机器表达式;如果可能的话,这应该看起来像一个有效的python表达式,可用于重新创建具有相同值的对象;如果无法做到这一点,则应返回形式上的字符串

object.__str__(self)

通过str(),format(),print()方法调用,计算对象的可打印字符串

object.__repr__()的不同在于,__str__()不要求返回有效的python表达式,即可以使用更方便更简洁的表示方式

object.__bytes__(self)

通过bytes调用,计算对象的字节码

object.__bool__(self)

返回True或False,如果此方法未定义,__len__()被调用,非0意味着True;如果2个方法都未定义,则认定返回值为True

2.2 基本属性访问

object.__getattr__(self, name)

当调用__getattribute__()方法抛出AttributeError异常,或者__get__()方法抛出AttributeError异常时,__getattr__()才被调用,可以返回某个值,异或同样抛出异常

object.__getattrbute__(self, name)

访问实例的属性时无条件调用

为了避免无限递归,方法内部在访问对象的属性时,应始终使用类方法,而不是A.a的形式

object.__setattr__(self, name, value)

属性赋值时调用

object.__delattr__(self, name)

删除对象的属性

object.__get__(self, instance, owner)

在获取类(owner)的属性,或者类的实例(instance)的属性时调用

object.__set__(self, instance, value)

将实例的属性设置为新值

object.__delete__(self, instance)

删除实例的属性

2.3 自定义类创建

object.__init_subclass__(cls)

当一个类继承自另一个类时,另一个类的__init_subclass__()方法都将被调用

class Philosopher:
def __init_subclass__(cls, default_name, **kwargs):
super().__init_subclass__(**kwargs)
cls.default_name = default_name class AustralianPhilosopher(Philosopher, default_name="Bruce"):
pass

默认情况下,object.__init_subclass__()无任何操作,但是被调用时,如果有传入参数,会抛出异常

元类(metaclass)

默认情况下,类对象通过type()函数创建,type(name, bases, namespace)

类对象的创建过程,可以通过传递metaclass关键字属性,或者继承自另一个拥有此参数的类

定义类对象时,会执行如下操作:

  • MRO entries are resolved(还不清楚具体含义)

    • 如果基类不是type,会搜索__mro_entries__方法;如果发现了,通过original bases tuple调用;该方法必须返回类的元组,当然可以为空
  • the appropriate metaclass is determined
    • 如果没有明确指定metaclass,则使用type
    • 如果指定了metaclass,且不是type的实例,那么直接使用
    • 如果指定了type的实例作为metaclass,那么将追溯到顶层的metaclass并使用
  • the class namespace is prepared
    • 如果metaclass有__prepare__属性,那么namespace = metaclass.__preprare__(name, bases, **kwargs)
    • 如果没有__prepare__属性,类的命名空间将被初始化为空的有序映射
  • the class body is executed
  • the class object is created
    • 通过执行类主体填充命名空间后,调用metaclass(name, bases, namespace, **kwargs)方法来创建类对象,额外的关键字参数与__prepare__相同

元类事例

class OrderedClass(type):

    @classmethod
def __prepare__(metacls, name, bases, **kwds):
return collections.OrderedDict() def __new__(cls, name, bases, namespace, **kwds):
result = type.__new__(cls, name, bases, dict(namespace))
result.members = tuple(namespace)
return result class A(metaclass=OrderedClass):
def one(self): pass
def two(self): pass
def three(self): pass
def four(self): pass >>> A.members
('__module__', 'one', 'two', 'three', 'four')

2.4 模拟可调用对象

object.__call__(self [, args...])

使实例可以像函数一样调用,假设定义方法x(arg1, arg2, ...),相当于调用x._call_(arg1, arg2, ...)`

2.5 模拟容器类型

object.__len__(self)

len()方法的实现,返回对象的长度

object.__getitem__(self, key)

self[key]的实现,对于序列,key必须为整数或切片对象

object.__missing__(self, key)

对字典类型数据,调用self[key]且key不在字典中时触发

object.__setitem__(self, key, value)

对self[key]赋值,注意只适用于key对应的值可以改变,或者可以追加新key

object.__delitem__(self, key)

删除self[key]

object.__iter__(self)

当容器需要迭代器时,调用此方法,返回一个新的迭代器对象

如果是映射类型,应当迭代容器的所有键

object.__reversed__(self)

reversed()方法的实现,返回一个新的迭代器,以倒序形式迭代容器中的元素

object.__contain__(self, item)

成员检测时调用,返回True或False

对于映射类型,只考虑键是否包含,而非值

2.6 with语句

上下文管理器是在执行with语句时定义要建立的运行时上下文的对象

上下文管理器处理对代码执行所需的运行时上下文的入口和出口

上下文管理器的典型应用包括,保存或恢复各种全局状态、锁定和解锁资源、打开关闭文件等

object.__enter__(self)

与对象相关的运行时上下文入口,with语句将方法的返回值绑定到as子句指定的目标

object.__exit__(self, exc_type, exc_value, traceback)

与对象相关的运行时上下文出口,如果退出时无异常,三个参数都为None

如果有异常出现,此方法希望禁止异常抛出,从而返回一个真正的值;否则异常将在退出此方法时正常处理

python 编程模型的更多相关文章

  1. 基于python yield机制的异步操作同步化编程模型

    又一个milestone即将结束,有了些许的时间总结研发过程中的点滴心得,今天总结下如何在编写python代码时对异步操作进行同步化模拟,从而提高代码的可读性和可扩展性. 游戏引擎一般都采用分布式框架 ...

  2. 学习笔记TF048:TensorFlow 系统架构、设计理念、编程模型、API、作用域、批标准化、神经元函数优化

    系统架构.自底向上,设备层.网络层.数据操作层.图计算层.API层.应用层.核心层,设备层.网络层.数据操作层.图计算层.最下层是网络通信层和设备管理层.网络通信层包括gRPC(google Remo ...

  3. Spark:Spark 编程模型及快速入门

    http://blog.csdn.net/pipisorry/article/details/52366356 Spark编程模型 SparkContext类和SparkConf类 代码中初始化 我们 ...

  4. spark概念、编程模型和模块概述

    http://blog.csdn.net/pipisorry/article/details/50931274 spark基本概念 Spark一种与 Hadoop 相似的通用的集群计算框架,通过将大量 ...

  5. 书籍推荐Python编程:从入门到实践(高清完整pdf)

    这本书我看了电子版的,感觉还不错,全书共有20章,书中的简介如下: 本书旨在让你尽快学会 Python ,以便能够编写能正确运行的程序 -- 游戏.数据可视化和 Web 应用程序,同时掌握让你终身受益 ...

  6. Python编程从入门到实践笔记——函数

    Python编程从入门到实践笔记——函数 #coding=gbk #Python编程从入门到实践笔记——函数 #8.1定义函数 def 函数名(形参): # [缩进]注释+函数体 #1.向函数传递信息 ...

  7. DataFlow编程模型与Spark Structured streaming

    流式(streaming)和批量( batch):流式数据,实际上更准确的说法应该是unbounded data(processing),也就是无边界的连续的数据的处理:对应的批量计算,更准确的说法是 ...

  8. arp协议分析&python编程实现arp欺骗抓图片

    arp协议分析&python编程实现arp欺骗抓图片 序 学校tcp/ip协议分析课程老师布置的任务,要求分析一种网络协议并且研究安全问题并编程实现,于是我选择了研究arp协议,并且利用pyt ...

  9. 3、TensorFlow基础(一) 设计思想与编程模型

    1.TensorFlow系统架构 如图为TensorFlow的系统架构图: TensorFlow的系统架构图,自底向上分为设备层和网络层.数据操作层.图计算层.API层.应用层,其中设备层和网络层,数 ...

随机推荐

  1. 03013_动态页面技术-JSP

    1.jsp的出现 2.jsp脚本和注释 (1)jsp脚本 ①<%java代码%> ----- 内部的java代码翻译到service方法的内部: ②<%=java变量或表达式> ...

  2. easyui的layout

    1.浏览器自适应(即浏览器改变大小,里面的表格大小也会随之改变)要设置两个参数 (1)一般都要在body上设置class=“easyui-layout”: <body class="e ...

  3. easyui datagrid复选框控制单选

    使用easyui datagrid的时候,由于对数据表格操作太多,并且有单选和多选功能因此采用复选框.但是在单选的状态,使用CheckOnSelect和singleselect时发现,页面有明显延迟, ...

  4. 使用vue+webpack的多页面架构(转+自己的情况)

    按以下步骤可正常配置多页面架构 记得安装 node-glob   安装命令:npm install node-glob --save-dev 文件附加 webpack.base.conf.js --参 ...

  5. 网络安全巧设置 Win2008 R2 防火墙详解(1)

    针对一般中小企业型来说,如果希望对企业网络进行安全管理,不一定非得花高价钱购买专业的防火墙设置,直接借助操作系统本身自带的防火墙功能即可以满足一般企业的应用,今天我们就一起来探究一下Windows S ...

  6. Leetcode 593.有效正方形

    有效正方形 给定二维空间中四点的坐标,返回四点是否可以构造一个正方形. 一个点的坐标(x,y)由一个有两个整数的整数数组表示. 示例: 输入: p1 = [0,0], p2 = [1,1], p3 = ...

  7. zookeeper 下载安装

    下载:wget https://www-us.apache.org/dist/zookeeper/zookeeper-3.4.13/zookeeper-3.4.13.tar.gz 解压:tar -zx ...

  8. BZOJ1797 [Ahoi2009]Mincut 最小割 【最小割唯一性判定】

    题目 A,B两个国家正在交战,其中A国的物资运输网中有N个中转站,M条单向道路.设其中第i (1≤i≤M)条道路连接了vi,ui两个中转站,那么中转站vi可以通过该道路到达ui中转站,如果切断这条道路 ...

  9. pdf生成

    要用本文的方法生成PDF文件,需要两个控件:itextsharp.dll和ICSharpCode.SharpZipLib.dll,由于示例代码实在太多,我将代码全部整理出来,放在另外一个文件“示例代码 ...

  10. readonly和disabled区别

    1.readonly属性只对表单元素的文本框.密码框和 textarea 有效,而disabled属性对所有的表单元素都会有效. 2.设置两个属性的外观不一样,这个自己可以观察一下. 3.设置read ...