一、基础知识

“__”前缀:私有属性、方法,在__dict__中存储时被改写为“_类名__”前缀

“_”前缀:是约定俗成的保护属性、方法,不过编译器不会对之采取任何处理

二、class特殊方法介绍

在第9章的双元素基础上,我们扩展为任意元素长度的向量,并探讨了相关方法的使用:

__getattr__(self, name):

result = obj.name 会调用builtin函数getattr(obj,'name')查找对应属性,如果没有name属性则调用obj.__getattr__('name')方法,再无则报错

__setattr__(self, name, value):

obj.name = value 会调用builtin函数setattr(obj,'name',value)设置对应属性,如果设置了__setattr__('name',value)方法则优先调用此方法,而非直接将值存入__dict__并新建属性

自己改写时一般添加一些预先处理,最后还是要用到继承super().__setattr__(name, value)

__getitem__(self, item):

在函数体内,会将self[index]的外部表达中的index转化为slice对象,方便处理

__setitem__(self, key, value):

没有用到,用于设置可以self[key]形式查询的数据,即支持序列索引和散列表两种形式

实现程序如下:

import math
import reprlib
from array import array
import numbers # 提供整数基类
import functools
import operator # 亦或方法,虽然'^='功能一样
import itertools class Vector:
typecode = 'd'
shortcut_names = 'xyzt' def __init__(self, components):
self._components = array(self.typecode, components) def __iter__(self):
return iter(self._components) def __repr__(self):
components = reprlib.repr(self._components)
components = components[components.find('['):-1]
return 'Vector({})'.format(components) def __str__(self):
return str(tuple(self)) def __bytes__(self):
return (bytes([ord(self.typecode)]) +
bytes(self._components)) def __abs__(self):
return math.sqrt(sum(x*x for x in self)) def __bool__(self):
return bool(abs(self)) @classmethod
def frombytes(cls, octets):
typecode = chr(octets[0])
memv = memoryview(octets[1:]).cast(typecode)
return cls(memv) # ==========支持切片操作==========
def __len__(self):
return len(self._components) # def __getitem__(self, index):
# # 支持切片类似[1:2]
# # 但是Vector的切片是数组,而非Vector实例
# # 实际上在方法内部index被转化为了slice对象
# return self._components[index] # 和__setitem_对应(此处未实现)
def __getitem__(self, index): # 会将self[index]的index转化为slice对象
cls = type(self)
if isinstance(index, slice):
return cls(self._components[index])
elif isinstance(index, numbers.Integral):
return self._components[index]
else:
raise TypeError('indices must be integers') # ==========动态存取属性==========
# xyzt四个字母分别指代前四个分量,调用如'v.x',
# 设置属性时保证不能使用小写字母a-z为实例添加属性
def __getattr__(self, name):
cls = type(self)
if len(name) == 1:
pos = cls.shortcut_names.find(name)
if 0 <= pos < len(self._components):
return self._components[pos]
raise AttributeError('object has no attribute') def __setattr__(self, name, value):
cls = type(self)
if len(name) == 1:
if name in cls.shortcut_names:
error = 'readonly attribute {attr_name!r}'
elif name.islower():
error = "can't set attributes 'a' to 'z' in {cls_name!r}"
else:
error = '' # bool('')=False
if error:
raise AttributeError(error.format(cls_name=cls.__name__,
attr_name=name))
super().__setattr__(name, value) # ==========哈希方法==========
def __hash__(self):
# hashes = map(hash, self._components)
hashes = (hash(x) for x in self._components)
return functools.reduce(operator.xor, hashes, initial=0)
# reduce的initial表示初始值,如果序列为空则返回初始值,否则
# 初始值加到序列首,对亦或运算初始值应该为0 def __eq__(self, other):
# ----------version1----------
# 原版(注释句)在向量很长时消耗内存过巨,需要构建两个元组
# return tuple(self) == tuple(other)
# ----------version2----------
# if len(self) != len(other):
# return False
# for a, b in zip(self, other):
# if a != b:
# return False
# return True
# ----------version3----------
return len(self) == len(other) and \
all(a == b for a, b in zip(self, other)) # format规范化输出
def angle(self, n):
r = math.sqrt(sum(x * x for x in self._components))
a = math.atan2(r, self[n-1])
if (n == len(self) - 1) and (self[-1] < 0):
return math.pi * 2 - a
else:
return a def angles(self):
return (self.angle(n) for n in range(1, len(self))) def __format__(self, format_spec=''):
if format_spec.endswith('h'):
format_spec = format_spec[:-1]
# r, angle1,angle2,……
coords = itertools.chain([abs(self)], self.angles())
out_fmt = '<{}>'
else:
coords = self
out_fmt = '({})'
components = (format(c, format_spec) for c in coords)
return out_fmt.format(','.join(components)) if __name__ == "__main__":
print(Vector(range(10)))
v = Vector(range(7))
print(v)
print(v[3])
print(v.x)

『流畅的Python』第10章笔记_序列类型的更多相关文章

  1. 『流畅的Python』第9章笔记_对象

    一.Python风格 以一个二元素向量对象为例 import math from array import array class Vector2d: typecode = 'd' def __ini ...

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

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

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

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

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

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

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

  8. 《流畅的Python》Data Structures--第2章序列array

    第二部分 Data Structure Chapter2 An Array of Sequences Chapter3 Dictionaries and Sets Chapter4 Text vers ...

  9. Python基本语法_基本数据类型_序列类型详解

    目录 目录 序列 序列的标准操作符 切片操作符 一个例子 字符串的连接 序列的功能函数 enumerate 枚举出序列对象的元素 len 获取序列对象的长度 min 取出sequence中的最小值 m ...

随机推荐

  1. mysql重要sql小记

    mysql -hip -uuser -p -A DB -e '[sql]' | sed 's/\t/","/g;s/^/"/;s/$/"/;s/\n//g' & ...

  2. 论文笔记:Variational Capsules for Image Analysis and Synthesis

    Variational Capsules for Image Analysis and Synthesis  2018-07-16 16:54:36 Paper: https://arxiv.org/ ...

  3. 【C#】可空类型 NullAble<T>

    在实际编写代码时候 ,  会遇到很多场景, 需要将值置成空, 比如发货日期, 有可能是没有. 在没有可空类型之前, 程序都是用 魔值, 即为一个minValue或者常量, 来代表这个值为空, 也有用一 ...

  4. 【ASP.Net】publish asp.net to local IIS

    做web项目难免要将项目部署, 要么部署在azure上,要么部署在本地, 使用IIS去host. 部署步骤很简单, 1. vs打开你的web项目, 项目名上面右键选择publish 2. 在弹出的pu ...

  5. 解决Maven管理项目update Maven时,jre自动变为1.5

    本文为博主原创,未经允许不得转载: 在搭建一个maven web项目时,项目已经按步骤搭建完好,之后项目上就报了一个错误. 在控制台看到错误提示如下:Dynamic Web Module 3.0 re ...

  6. 《机器学习实战》之k-近邻算法(改进约会网站的配对效果)

    示例背景: 我的朋友海伦一直使用在线约会网站寻找合适自己的约会对象.尽管约会网站会推荐不同的人选,但她并不是喜欢每一个人.经过一番总结,她发现曾交往过三种类型的人: (1)不喜欢的人: (2)魅力一般 ...

  7. BZOJ 1070: [SCOI2007]修车(费用流)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1070 题意: 思路: 神奇的构图. 因为排在后面的人需要等待前面的车修好,这里将每个技术人员拆成n个 ...

  8. vscode已有64位版本。

    我的操作系统是win10 Family版本. vscode不知道什么鬼,只要开启没动任何操作,cpu就占到30%. 于是我打开任务管理器,选中vscode进程->转到详细信息->结束cpu ...

  9. 语法对照表ES5VSES6

    模块 导入 在ES5里面,如果使用CommonJS的标准,引入包一般是使用require来的 //ES5 js var React = require("react") var { ...

  10. React native中的组建通知通信:

    有这么一个需求,在B页面pop()回到A页面,需要A页面执行刷新,那么我们可以采用以下方法: 1:在A页面Push到B页面中,加上一个A页面中的刷新函数做为参数,然后在B页面中在pop()函数封装后通 ...