一、Python风格

以一个二元素向量对象为例

import math
from array import array class Vector2d:
typecode = 'd' def __init__(self, x, y):
self.x = float(x)
self.y = float(y) def __iter__(self):
# 使得Vector2d变成可迭代对象
# __iter__方法的实现使得本类可以被转化为tuple在内的其他可迭代类
return (i for i in (self.x, self.y)) def __repr__(self):
class_name = type(self).__name__ # type(self): <class '__main__.Vector2d'>
return '{}({!r},{!r})'.format(class_name, *self) def __str__(self):
return str(tuple(self)) def __eq__(self, other):
return tuple(self) == tuple(other) def __abs__(self):
return math.hypot(self.x, self.y) def __bool__(self):
return bool(abs(self)) def __bytes__(self):
"""将Vector2d对象处理为二进制序列,格式我们自定"""
# d:double类型数组
return (bytes([ord(self.typecode)]) +
bytes(array(self.typecode, self))) # —————备用析构方法——————
@classmethod # 类方法,cls表示类本身
def frombytes(cls, octets):
"""对应于上面的方法,这里创建一个新的析构函数,使用特定的二进制序列构造Vector2d类实例"""
typecode = chr(octets[0])
memv = memoryview(octets[1:]).cast(typecode)
return cls(*memv) # 类名(参数),可见,类方法常用作备用析构 # —格式化输出—
def angle(self):
# math.atan(scope)输入为tan值
# math.atan2(y, x)输入为对应向量坐标(起点为原点)
return math.atan2(self.y, self.x) def __format__(self, fmt_spec=''):
"""格式化输出,如果格式末尾为p则输出极坐标,
输入其他格式为数字型格式,一个输入格式指定到两个数上,如:.3ep"""
if fmt_spec.endswith('p'):
fmt_spec = fmt_spec[:-1]
coords = (abs(self), self.angle())
out_fmt = '<{}, {}>'
else:
coords = self
out_fmt = '({}, {})'
components = (format(c, fmt_spec) for c in coords)
return out_fmt.format(*components)

此时这个对象支持大部分python操作,

if __name__ == '__main__':
b = bytes(Vector2d(3, 4))
print(Vector2d.frombytes(b)) print(format(Vector2d(1, 1), '.5fp'))

(3.0, 4.0)
<1.41421, 0.78540>

但是一个重要的方法还是没能实现,__hash__,这关乎到对象是否可以被存入字典进行高速读取的属性,实际上可以hash对象需要三个条件:

  1. 需要__hash__方法
  2. 需要__eq__方法(已经实现)
  3. 需要对象不可变  # 实例的散列值关乎查找等使用方式,绝对不可以变化

也就是我们指定v.x=1(v为class实例)会报错才行,这需要一些其他操作:

class Vector2d:
typecode = 'd' def __init__(self, x, y):
self.__x = float(x)
self.__y = float(y) @property
def x(self):
return self.__x @property
def y(self):
return self.__y def __hash__(self):
return hash(self.x) ^ hash(self.y)  

其他方法不需要修改,

v1 = Vector2d(3, 4)

v2 = Vector2d(3.1, 4.2)

print(hash(v1), hash(v2))

# 7 384307168202284039

二、类方法和静态方法

# —对比类方法和静态方法—
class Demo:
@classmethod
def klassmeth(*args):
return args @ staticmethod
def statmeth(*args):
return args def normal(*args):
return args

和实例方法不同,类方法第一个参数永远是类本身,所以常用于备用析构,静态方法没有默认的首位参数,测试如下:

print(Demo.klassmeth("hello"))
print(Demo.statmeth("hello"))
demo = Demo()
print(demo.normal("hello"))

# (<class '__main__.Demo'>, 'hello')
# ('hello',)
# (<__main__.Demo object at 0x000000000289F978>, 'hello')

三、私有属性和受保护属性

两个前导下划线"__",一个或者没有后置下划线的实例属性(self.属性名)为私有变量,会被存入__dict__中,且名称被改写为"_类名__属性名",主要目的是防止类被继承以后,子类实例的继承属性(未在子类中显式的声明)被错误的改写。

注意,__dict__不仅仅存储私有变量,实例属性均存放在__dict__中(默认情况下)。

四、__slots__类属性节约存储空间

class Vector2d:

    __slots__ = ('__x', '__y')

    typecode = 'd'

类属性__slots__为一个存储字符串的可迭代对象,其中的各个字符串是不同的实例属性名,使用tuple是作者推荐的方式,因为可以保证信息不被改动。使用它可以有效节约存储空间,尤其是需要创建大量实例的时候(运行速度往往也更快)。

  1. 继承会自动忽略__slot__属性,所以子类需要显式的定义它
  2. 定义了__slots__后,用户不可以自行添加其他实例属性,但是如果把__dict__存储在__slots__中,就可以添加了,不过就完全没意义了……
  3. 如果想要支持弱引用,需要手动将__weakref__添加进来,虽然自定义class默认存在__weakref__属性,但是想要让实例成为弱引用目标还是需要添加进来才可以。

五、覆盖类属性

实例调用的实例属性时,如果属性不存在,会去读取同名的类属性(例子中的typecode类属性被self.typecode调用)

实例新建实例属性时,如果已经存在同名的类属性,不会改写类属性,而会对实例新建一个实例属性

『流畅的Python』第9章笔记_对象的更多相关文章

  1. 『流畅的Python』第10章笔记_序列类型

    一.基础知识 “__”前缀:私有属性.方法,在__dict__中存储时被改写为“_类名__”前缀 “_”前缀:是约定俗成的保护属性.方法,不过编译器不会对之采取任何处理 二.class特殊方法介绍 在 ...

  2. 『流畅的Python』第5章笔记_一等函数

  3. 『流畅的Python』第15章:上下文管理器和else块

  4. 『流畅的Python』第14章:可迭代的对象、迭代器和生成器

  5. 『流畅的Python』第13章:正确重载运算符

  6. 『流畅的Python』第12章:继承的优缺点

  7. 『流畅的Python』第1~4章笔记_数据结构、编码

    由于1~4章内容零散且基础,所以统计一下涉及到的内容,记录一下,方便查阅(第一张图右键新页面打开即可看到清晰大图)

  8. 流畅的python第十四章可迭代的对象,迭代器和生成器学习记录

    在python中,所有集合都可以迭代,在python语言内部,迭代器用于支持 for循环 构建和扩展集合类型 逐行遍历文本文件 列表推导,字典推导和集合推导 元组拆包 调用函数时,使用*拆包实参 本章 ...

  9. 《流畅的Python》Data Structures--第7章 colsure and decorator

    Function Decorators and Closures 装饰器是用于增强函数的行为,理解它,就必须先理解闭包. Python3引入关键字nonlocal,如果要理解闭包,就必须了解它的所有方 ...

随机推荐

  1. IDEA入门及maven配置

    idea基本使用 下载安装 idea下载地址 激活方法 上面的失效使用这个 配置SDKs和编译版本 选择jdk 我是主要用于Java开发,因此配置JDK8 配置编码 其他配置 个人配置主题等 注意 与 ...

  2. 论文阅读之:Deep Meta Learning for Real-Time Visual Tracking based on Target-Specific Feature Space

    Deep Meta Learning for Real-Time Visual Tracking based on Target-Specific Feature Space  2018-01-04  ...

  3. (转)PaperWeekly 第二十二期---Image Caption任务综述

    本文转自:http://mp.weixin.qq.com/s?__biz=MzIwMTc4ODE0Mw==&mid=2247484014&idx=1&sn=4a053986f5 ...

  4. Unity3D学习笔记(三十六):Shader着色器(3)- 光照

    光照模型:用数学的方法模拟现实世界中的光照效果.   场景中模型身上的光反射到相机中的光线: 1.漫反射:产生明暗效果 2.高光反射:产生镜面反射,物体中有最亮且比较耀眼的一部分 3.自发光: 4.环 ...

  5. Python 编码规范 PEP8

    1 Introduction Guido 的核心思想是:对于代码而言,相比于写,它更多是被用来读的.这个指导旨在使Python代码更易读,且具有更强的协调性. 2 A Foolish Consiste ...

  6. SAP FI 常用表

    SAP FI 常用表 GL 部分: FAGLFLEXT 新总账汇总表 GLT0 旧总帐汇总表 SKA1 总账科目主记录 (科目表) 科目表层数据 SKAT 总帐科目主记录(科目表:说明) 包括语言代码 ...

  7. 解决github网站打开慢的问题

    一.前言 作为一名合格的程序员,github打开速度太慢怎么能容忍.但是可以通过修改hosts文件信息来解决这个问题.现在chrome访问github速度杠杠的! 二.macOS解决方法 打开host ...

  8. _ai_gameobject

  9. 1. eclipse异常问题解决办法

    1. 内存溢出问题 问题描述:报错信息 java.lang.OutOfMemoryError: PermGen space 解决办法: 在Tomcat/bin/catalina.bat 文件下加入: ...

  10. java扫描文件夹下面的所有文件(递归与非递归实现)

    java中扫描指定文件夹下面的所有文件扫描一个文件夹下面的所有文件,因为文件夹的层数没有限制可能多达几十层几百层,通常会采用两种方式来遍历指定文件夹下面的所有文件.递归方式非递归方式(采用队列或者栈实 ...