Python :数据模型

news2025/3/14 10:41:54

一. 什么是数据模型?

Python数据模型是Python对象系统的抽象,通过一组特殊方法​(如__init____len__等)和协议​(如迭代协议、上下文管理协议),定义了对象如何与语言的内置功能(如len()for循环等)交互。

核心思想

  • 统一性:所有对象(如列表、字典、自定义类)的行为都通过相同的特殊方法实现。
  • 灵活性:通过实现特殊方法,可以让自定义对象支持内置操作(如+in、切片等)。

例子

import collections
Card = collections.namedtuple('Card', ['rank', 'suit'])
class FrenchDeck:
    ranks = [str(n) for n in range(2, 11)] + list('JQKA')
    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, position):
        return self._cards[position]
解释
  1. 模块导入

    import collections

    导入了 Python 标准库中的 collections 模块。

  2. 命名元组定义:Card

    Card = collections.namedtuple('Card', ['rank', 'suit'])

    使用 namedtuple 创建了一个名为 Card 的简单类,表示扑克牌的一张牌。每个 Card 对象有两个属性:

    • rank:表示牌的点数(如 '2''J''A' 等)。
    • suit:表示牌的花色(如 'spades''diamonds' 等)。
  3. 类定义:FrenchDeck

    • 定义了一个名为 FrenchDeck 的类,表示一副标准的 52 张扑克牌。
    • 类属性:

      ranks = [str(n) for n in range(2, 11)] + list('JQKA') suits = 'spades diamonds clubs hearts'.split()

      • ranks:表示牌的所有点数,包括数字牌('2' 到 '10')和字母牌('J''Q''K''A')。
      • suits:表示牌的所有花色('spades' 黑桃、'diamonds' 方片、'clubs' 梅花、'hearts' 红心)。
    • 构造函数 __init__

      def __init__(self): self._cards = [Card(rank, suit) for suit in self.suits for rank in self.ranks]

      初始化时创建了一副完整的扑克牌,存储在 _cards 列表中。通过列表推导式生成所有可能的牌组合。
    • 方法 __len__

      def __len__(self): return len(self._cards)

      实现了特殊方法 __len__,使得可以通过 len(deck) 获取扑克牌的数量。
    • 方法 __getitem__

      def __getitem__(self, position): return self._cards[position]

      实现了特殊方法 __getitem__,使得可以通过索引访问扑克牌,例如 deck[0]

常用特殊方法

方法对应操作
__init__对象初始化
__repr__ / __str__字符串表示
__len__len(obj)
__getitem__obj[key](支持索引和切片)
__iter__for x in obj(迭代)
__enter__ / __exit__with语句(上下文管理)

迭代协议

实现__iter____next__方法,让对象支持for循环:

class CountDown:
    def __init__(self, start):
        self.current = start

    def __iter__(self):
        return self

    def __next__(self):
        if self.current <= 0:
            raise StopIteration
        else:
            self.current -= 1
            return self.current + 1

for num in CountDown(3):
    print(num)  # 输出: 3 2 1

 

 解释
  1. 类定义:CountDown

    • 定义了一个名为 CountDown 的类,用于实现倒计时功能。

    • 构造函数 __init__

      def __init__(self, start): self.current = start

      初始化时接收一个参数 start,表示倒计时的起始值,并将其赋值给实例属性 self.current

    • 方法 __iter__

      def __iter__(self): return self

      实现了可迭代协议,使得该类的实例可以作为迭代器使用。返回自身(self)。

    • 方法 __next__

      def __next__(self): if self.current <= 0: raise StopIteration else: self.current -= 1 return self.current + 1

      实现了迭代器协议中的 __next__ 方法:

      • 如果当前值 self.current 小于或等于 0,则抛出 StopIteration 异常,表示迭代结束。
      • 否则,将 self.current 减 1,并返回减 1 前的值(即当前值)。
  2. 使用场景

    • 该类可以通过 for 循环或手动调用 next() 方法进行倒计时。
    • 示例:
      countdown = CountDown(5)
      for value in countdown:
          print(value)
      输出结果:

      5 4 3 2 1

上下文管理协议

实现__enter____exit__方法,支持with语句:

class Timer:
    def __enter__(self):
        self.start = time.time()
        return self

    def __exit__(self, exc_type, exc_val, exc_tb):
        self.end = time.time()
        print(f"耗时: {self.end - self.start:.2f}秒")

with Timer():
    time.sleep(1)  # 输出: 耗时: 1.00秒
 解释
  1. 类定义:Timer

    • 定义了一个名为 Timer 的类,用于测量代码块的执行时间。

    • 方法 __enter__

      def __enter__(self): self.start = time.time() return self

      • 实现了上下文管理协议中的 __enter__ 方法。
      • 在进入 with 语句块时,记录当前时间(以秒为单位)并存储在实例属性 self.start 中。
      • 返回自身(self),以便在 with 语句块中可以访问该对象。
    • 方法 __exit__

      def __exit__(self, exc_type, exc_val, exc_tb): self.end = time.time() print(f"耗时: {self.end - self.start:.2f}秒")

      • 实现了上下文管理协议中的 __exit__ 方法。
      • 在退出 with 语句块时,记录当前时间(以秒为单位)并存储在实例属性 self.end 中。
      • 计算执行时间(self.end - self.start),并以两位小数的格式打印到控制台。
  2. 使用场景

    • 通过 with 语句使用 Timer 类来测量代码块的执行时间。
    • 示例:

      with Timer(): time.sleep(1) # 输出: 耗时: 1.00秒

      • 在 with 语句块中调用了 time.sleep(1),模拟了一个耗时 1 秒的操作。
      • Timer 类会自动计算并打印出这段代码的执行时间。 

让自定义类支持切片

class MySequence:
    def __init__(self, data):
        self.data = data

    def __len__(self):
        return len(self.data)

    def __getitem__(self, index):
        if isinstance(index, slice):
            return self.data[index.start : index.stop : index.step]
        else:
            return self.data[index]

seq = MySequence([0, 1, 2, 3, 4])
print(seq[1:3])  # [1, 2](支持切片!)

鸭子类型(Duck Typing)​

Python不检查对象的类型,而是检查对象是否实现了特定的方法(即是否“像鸭子一样叫”)。

示例
class FakeList:
    def __len__(self):
        return 10

    def __getitem__(self, index):
        return index * 2

fake = FakeList()
print(len(fake))       # 10(调用__len__)
print(fake[5])         # 10(调用__getitem__)
print(isinstance(fake, list))  # False,但行为类似列表!

 

 

 

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2314812.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

idea超级AI插件,让 AI 为 Java 工程师

引言​ 用户可在界面中直接通过输入自然语言的形式描述接口的需求&#xff0c;系统通过输入的需求自动分析关键的功能点有哪些&#xff0c;并对不确定方案的需求提供多种选择&#xff0c;以及对需求上下文进行补充&#xff0c;用户修改确定需求后&#xff0c;系统会根据需求设…

µCOS-III从入门到精通 第十四章(软件定时器)

参考教程&#xff1a;【正点原子】手把手教你学UCOS-III实时操作系统_哔哩哔哩_bilibili 一、软件定时器简介 1、定时器的概念与种类 &#xff08;1&#xff09;定时器的概念&#xff1a;从指定的时刻开始&#xff0c;经过一个指定时间&#xff0c;然后触发一个超时事件&…

MySQL数据库复杂的增删改查操作

在前面的文章中&#xff0c;我们主要学习了数据库的基础知识以及基本的增删改查的操作。接下去将以一个比较实际的公司数据库为例子&#xff0c;进行讲解一些较为复杂且现时需求的例子。 基础知识&#xff1a; 一文清晰梳理Mysql 数据库基础知识_字段变动如何梳理清楚-CSDN博…

KCD 北京站丨Volcano 邀您畅聊云原生智能调度技术与应用

AI与云原生技术正以前所未有的速度改变着我们的世界&#xff0c;而云原生技术则如同一座坚实的桥梁&#xff0c;连接着传统IT与现代化的数字世界。当AI与云原生相遇&#xff0c;它们相互赋能&#xff0c;相互促进&#xff0c;为开发者们打开了一个全新的技术宇宙。 3 月 15 日&…

BLEU评估指标

一、介绍 用于评估模型生成的句子和实际句子差异的指标&#xff0c;取值在[0,1]&#xff0c;匹配度高就距离1近&#xff0c;反之距离0近。这个指标计算代价小&#xff0c;容易理解&#xff0c;与语言无关&#xff0c;与人类评价结果高度相关。 BLEU主要基于n-gram匹配&#x…

高效自动化测试:打造Python+Requests+Pytest+Allure+YAML的接口测试框架

一、背景 在快节奏的开发周期中&#xff0c;如何确保接口质量&#xff1f;自动化测试是关键。通过构建标准化、可复用的测试框架&#xff0c;能显著提升测试效率与准确性&#xff0c;为项目质量保驾护航[1][7]。 二、目标 ✅ 核心目标&#xff1a; ● 实现快速、高效的接口测试…

BSides Vancouver: 2018 (Workshop)

BSides Vancouver: 2018 (Workshop) 来自 <https://www.vulnhub.com/entry/bsides-vancouver-2018-workshop,231/> 1&#xff0c;将两台虚拟机网络连接都改为NAT模式 2&#xff0c;攻击机上做namp局域网扫描发现靶机 nmap -sn 192.168.23.0/24 那么攻击机IP为192.168.23…

rStar论文精读

论文简介 论文标题&#xff1a;《Mutual reasoning makes smaller LLMs stronger problem-solvers》 论文地址&#xff1a;https://arxiv.org/abs/2408.06195 录用会议&#xff1a;ICLR2025 背景与挑战 挑战1&#xff1a;在SLM中平衡exploration与exploitation。一些方法有很…

247g 的工业级电调,如何让无人机飞得更 “聪明“?——STONE 200A-M 深度测评

一、轻量化设计背后的技术取舍 当拿到 STONE 200A-M 时&#xff0c;247g 的重量让人意外 —— 这个接近传统 200A 电调 70% 的重量&#xff0c;源自 1205624.5mm 的紧凑结构&#xff08;0.1mm 公差控制&#xff09;。实测装机显示&#xff0c;相比同规格产品&#xff0c;其体积…

Node.js:快速启动你的第一个Web服务器

Node.js 全面入门指南 文章目录 Node.js 全面入门指南一 安装Node.js1. Windows2. MacOS/Linux 二 配置开发环境1. VSCode集成 三 第一个Node.js程序1. 创建你的第一个Node.js程序 四 使用Express框架1. 快速搭建服务器 一 安装Node.js 1. Windows 以下是Windows环境下Node.j…

自定义日志回调函数实现第三方库日志集成:从理论到实战

一、应用场景与痛点分析 在开发过程中&#xff0c;我们经常会遇到以下场景&#xff1a; 日志格式统一&#xff1a;第三方库使用自己的日志格式&#xff0c;导致系统日志混杂&#xff0c;难以统一管理和分析。日志分级过滤&#xff1a;需要动态调整第三方库的日志输出级别&…

Linux练级宝典->任务管理和守护进程

任务管理 进程组概念 每个进程除了进程ID以外&#xff0c;还有一个进程组&#xff0c;进程组就是一个或多个进程的集合 同一个进程组&#xff0c;代表着他们是共同作业的&#xff0c;可以接收同一个终端的各种信号&#xff0c;进程组也有其唯一的进程组号。还有一个组长进程&a…

C语言:计算并输出三个整数的最大值 并对三个数排序

这是《C语言程序设计》73页的思考题。下面分享自己的思路和代码 思路&#xff1a; 代码&#xff1a; #include <stdio.h> int main() {int a,b,c,max,min,mid ; //设置大中小的数分别为max&#xff0c;mid&#xff0c;min&#xff0c;abc为输入的三个数printf("ple…

工具(十二):Java导出MySQL数据库表结构信息到excel

一、背景 遇到需求&#xff1a;将指定数据库表设计&#xff0c;统一导出到一个Excel中&#xff0c;存档查看。 如果一个一个弄&#xff0c;很复杂&#xff0c;耗时长。 二、写一个工具导出下 废话少絮&#xff0c;上码&#xff1a; 2.1 pom导入 <dependency><grou…

ACL初级总结

ACL–访问控制列表 1.访问控制 在路由器流量流入或者流出的接口上,匹配流量,然后执行相应动作 permit允许 deny拒绝 2.抓取感兴趣流 3.ACL匹配规则 自上而下逐一匹配,若匹配到了则按照对应规则执行动作,而不再向下继续匹配 思科:ACL列表末尾隐含一条拒绝所有的规则 华为:AC…

调优案例一:堆空间扩容提升吞吐量实战记录

&#x1f4dd; 调优案例一&#xff1a;堆空间扩容提升吞吐量实战记录 &#x1f527; 调优策略&#xff1a;堆空间扩容三部曲 # 原配置&#xff08;30MB堆空间&#xff09; export CATALINA_OPTS"$CATALINA_OPTS -Xms30m -Xmx30m"# 新配置&#xff08;扩容至120MB&am…

C语言 —— 此去经年梦浪荡魂音 - 深入理解指针(卷一)

目录 1. 内存和地址 2. 指针变量和地址 2.1 取地址操作符&#xff08;&&#xff09; 2.2 指针变量 2.3 解引用操作符 &#xff08;*&#xff09; 3. 指针的解引用 3.1 指针 - 整数 3.2 void* 指针 4. const修饰指针 4.1 const修饰变量 4.2 const修饰指针变量 5…

计算机毕业设计:留守儿童的可视化界面

留守儿童的可视化界面mysql数据库创建语句留守儿童的可视化界面oracle数据库创建语句留守儿童的可视化界面sqlserver数据库创建语句留守儿童的可视化界面springspringMVChibernate框架对象(javaBean,pojo)设计留守儿童的可视化界面springspringMVCmybatis框架对象(javaBean,poj…

golang算法二叉树对称平衡右视图

100. 相同的树 给你两棵二叉树的根节点 p 和 q &#xff0c;编写一个函数来检验这两棵树是否相同。 如果两个树在结构上相同&#xff0c;并且节点具有相同的值&#xff0c;则认为它们是相同的。 示例 1&#xff1a; 输入&#xff1a;p [1,2,3], q [1,2,3] 输出&#xff1a…

Chatbox通过百炼调用DeepSeek

解决方案链接&#xff1a;评测&#xff5c;零门槛&#xff0c;即刻拥有DeepSeek-R1满血版 方案概览 本方案以 DeepSeek-R1 满血版为例进行演示&#xff0c;通过百炼模型服务进行 DeepSeek 开源模型调用&#xff0c;可以根据实际需求选择其他参数规模的 DeepSeek 模型。百炼平台…