流畅的Python 1. Python数据模型(特殊方法 __func__())

文章目录

  • 1. `__getitem__()、__len__()` 方法
  • 2. 特殊方法

1. __getitem__()、__len__() 方法

  • 举一个扑克牌的例子
import collectionsCard = collections.namedtuple('Card_name', ['rank', 'suit'])print(Card.__doc__) # Card_name(rank, suit)class FrenchDeck:    ranks = [str(n) for n in range(2, 11)]   list('JKQA')    suits = 'spades diamonds clubs hearts'.split()    def __init__(self):        self._cards = [Card(rank, suit) for suit in self.suits         for rank in self.ranks]    def __len__(self):        return len(self._cards)    def __getitem__(self, pos):        return self._cards[pos]deck = FrenchDeck()print(len(deck))  # 13*4=52print(deck[0])  # [] 调用 __getitem__ 方法# Card_name(rank='2', suit='spades')from random import choiceprint(choice(deck))  # Card_name(rank='4', suit='clubs') 随机print(choice(deck))  # Card_name(rank='J', suit='diamonds')print(choice(deck))  # Card_name(rank='3', suit='clubs')# __getitem__ 支持切片操作# 取出前 3 个print(deck[:3])# 取出 A的所有项print(deck[12::13])  # 12 开始 结束没有写默认结尾,每隔13个取一次# __getitem__ 反向迭代也可以for card in reversed(deck):    print(card)# 没有实现 __contains__ 方法,in 顺序做一次迭代搜索print(Card('Q', 'hearts') in deck)  # Trueprint(Card('7', 'abc') in deck)  # False# 排序suit_values = dict(spades=3, hearts=2, diamonds=1, clubs=0)def spades_high(card):    # 自定义牌的 rank_value    rank_value = FrenchDeck.ranks.index(card.rank)    return rank_value*len(suit_values)   suit_values[card.suit]# 自定义排序for card in sorted(deck, key=spades_high):    print(card)

2. 特殊方法

  • python解释器会自动调用,如 len(obj) ,解释器 调用 obj__len__() 方法
  • 内置的类型的 __len__() 方法,抄近路,直接读取 ob_size 属性
  • for i in x:,是 iter(x) 调用了 x.__iter__() 方法
  • 通过内置函数(如,len,iter,str等)来使用特殊方法是最好的选择
  • 不要自己随意添加特殊方法 __func__ ,将来的 python版本 可能添加该方法

特殊方法:https://docs.python.org/3/reference/datamodel.html

from math import hypot  # 返回模长 Euclidean distanceclass Vector:    def __init__(self, x=0, y=0):        self.x = x        self.y = y    def __repr__(self):  # 打印的时候输出内容        print("调用__repr__")        # %r 获取 各个属性的标准字符串表示形式        return 'Vector(%r, %r)' % (self.x, self.y)        # __str__() 是在 str() 或 print() 时调用,返回的字符串更友好        # 请优先使用 __repr__, 如果调用 str() 但是有没有实现 __str__(),        # 解释器自动用 __repr__ 代替    # def __str__(self):    #     return "print() 优先调用 __str__()"    def __abs__(self):        print("调用__abs__")        return hypot(self.x, self.y)    def __bool__(self):        print("调用__bool__")        # bool(x) 调用 x.__bool__(), 如果不存在,则调用 x.__len__()        return bool(abs(self))        # 或者        # return (self.x or self.y) # 更高效    def __add__(self, other):        print("调用__add__")        x = self.x   other.x        y = self.y   other.y        # 原则:不改变输入,创建新的输出        return Vector(x, y)    def __mul__(self, scalar):        print("调用__mul__")        # 原则:不改变输入,创建新的输出        return Vector(self.x * scalar, self.y * scalar)    def __rmul__(self, scalar):  # 交换律        print("调用__rmul__")        # 原则:不改变输入,创建新的输出        return Vector(self.x * scalar, self.y * scalar)v1 = Vector(2, 4)v2 = Vector(2, 1)v3 = v1   v2  # 调用__add__print(v3)  # 调用__repr__  Vector(4, 5)print(abs(v3))  # 调用__abs__  6.4031242374328485print(bool(v3))  # 调用__bool__  调用__abs__  Trueprint(v3 * 3)  # 调用__mul__  调用__repr__  Vector(12, 15)print(3 * v3)  # 调用__rmul__ 调用__repr__  Vector(12, 15)print(str(v3))  # 调用__repr__  Vector(4, 5)

来源:https://www.icode9.com/content-1-883601.html

(0)

相关推荐

  • 为什么继承 Python 内置类型会出问题?!

    不久前,Python猫 给大家推荐了一本书<流畅的Python>(点击可跳转阅读),那篇文章有比较多的"溢美之词",显得比较空泛-- 但是,<流畅的Python& ...

  • 小白学PyTorch | 3 浅谈Dataset和Dataloader

    人人都可以玩转大数据 阿里云天池推荐 作者:机器学习炼丹术  来源:机器学习炼丹术 文章目录: 1 Dataset基类 2 构建Dataset子类 2.1 __Init__ 2.2 __getitem ...

  • Python编程基础:序列类型概述

    https://m.toutiao.com/is/eYtBNf7/ 序列首先是一种数据存储方式,用来存储一系列的数据.序列存储数据的主要特点就是数据在内存空间中是连续存储的,例如字符串abc(字符串属 ...

  • 第41天:Python operator 模块

    operator模块 operator 模块提供了一套与 Python 的内置运算符对应的高效率函数. 函数的种类 函数包含的种类有:对象的比较运算.逻辑运算.数学运算和序列运算 比较运算 运算 函数 ...

  • 第一次把 Python 的切片理解得如此透彻

    来源:Python猫 作者:豌豆花下猫 众所周知,我们可以通过索引值(或称下标)来查找序列类型(如字符串.列表.元组-)中的单个元素,那么,如果要获取一个索引区间的元素该怎么办呢? 切片(slice) ...

  • PyTorch 学习笔记(一):让PyTorch读取你的数据集

    加入极市专业CV交流群,与6000+来自腾讯,华为,百度,北大,清华,中科院等名企名校视觉开发者互动交流!更有机会与李开复老师等大牛群内互动! 同时提供每月大咖直播分享.真实项目需求对接.干货资讯汇总 ...

  • Python|二叉树叶子结点问题解决方法

    问题描述键盘输入一颗二叉树,求解其叶子结点个数.示例: 输入:4,2,6,1,3,5输出:3解决方案一棵树当中没有子结点(即度为0)的结点称为叶子结点,简称"叶子".当二叉树为空时 ...

  • Python|二叉树的遍历问题解决方法

    问题描述二叉树是由n个结点的有限集合,该集合或者为空集,或者由一个根节点和两颗互不相交的.分别称为根节点的左子树和右子树的二叉树组成.二叉树特征:每个结点最多只有两颗子树,即二叉树中结点的度最高不能超 ...

  • Selenium2+python自动化45-18种定位方法(find_elements)

    前言 江湖传言,武林中流传八种定位,其中xpath是宝刀屠龙,css是倚天剑. 除了这八种,其实还有十种定位方法,眼看就快失传了,今天小编让失传已久的定位方法重出江湖! 一.十八种定位方法 前八种是大 ...

  • Common encryption methods and implementation in Python Python中常用的加密方法及实现

    Common encryption methods and implementation in Python Python中常用的加密方法及实现

  • Python 下载文件的多种方法

    来源:Python 技术「ID: pythonall」 本文档介绍了 Python 下载文件的各种方式,从下载简单的小文件到用断点续传的方式下载大文件. Requests 使用 Requests 模块 ...

  • 列表推导式:简洁高效更具 Python 风格的列表创建方法

    我们在<Python 中的列表和元组>中已经详细介绍了列表(list)的基本特性和使用方法,本文将着重介绍一种 Python 中用于创建 list 的简洁高效的语法形式:列表推导式. Py ...

  • 一文讲解Python条件语句的使用方法

    Python条件语句是由执行一条或多条语句的结果(True或False)来决定要执行的代码块.Python程序语言指定true和非空(null)值均为true,0或null为false. Python ...

  • Python读取Excel文件的方法

    方法一:读excel文件单元格数据 import xlrd book = xlrd.open_workbook('fruit.xlsx') print('sheet页名称:',book.sheet_n ...

  • Python tkinter之控件方法bind的使用

    Python tkinter之控件方法bind的使用