Python中容易被忽略的内置类型

news/发布时间2024/5/10 23:41:57

Python中的内置类型是我们开发中最常见的,很多人都能熟练的使用它们。

然而有一些内置类型确实不那么常见的,或者说往往会被我们忽略,所以这次的主题就是带领大家重新认识这些“不同寻常”的内置类型。

1.frozenset

不可变集合(frozenset)与普通的set一样,只不过它的元素是不可变的,因此诸如addremoveupdate等可以添加/删除/改变集合内元素的方法是不存在的,换句话说一旦frozenset建立后你将不再可能更改集合内的元素。

其他的方法与set一致:

>>> frozen = frozenset([1, 1, 2, 3, 4, 5, 6, 6]) 
frozenset({1, 2, 3, 4, 5, 6}) 
>>> frozen | {1, 2, 3, 7, 8} 
frozenset({1, 2, 3, 4, 5, 6, 7, 8}) 
>>> frozen ^ {1, 2, 3, 7, 8} 
frozenset({4, 5, 6, 7, 8})

2.range

range事实上相当得常见,所以你也许会奇怪我为什么把它列出来。

其实原因很简单,因为大部分人熟悉range的使用,但并不清楚range到底是什么。返回迭代器?返回一个可迭代对象?range本身又是什么呢?

答案揭晓:

>>> range<class 'range'>

是的,range是个class!所以当我们使用for i in range(1, 10)这样的代码时,实际上我们遍历了一个range对象,而range也实现了可迭代对象需要的__iter__魔法方法,所以它自身是可迭代对象:

>>> range.__iter__<slot wrapper '__iter__' of 'range' objects>

因此,range既不返回迭代器,也不返回其他可迭代对象,而是返回的自己。

3.bytearray

bytearray一般情况下并不常见,它主要为了可以实现原地修改bytes对象而出现,因为bytes和str一样是不可变对象,例如这样是非法的:

>>> b = '测试用例a'.encode('utf8') 
>>> b[-1] = 98 # change 'a' -> 'b' 
Traceback (most recent call last): File "", line 1, in TypeError: 'bytes' object does not support item assignment 

而当我们把bytes的内容复制给bytearray时就可以进行原地修改了:

>>> array = bytearray(b) 
>>> array[-1] = 98 
>>> array.decode('utf8') 

bytearray对象没有字面常量,因此只能通过构造函数创建,它有着和bytes一样的方法,只是可变以及多了一些序列对象的特性。如果要创建一个bytearray可以有如下的几种方法: - bytearray()返回一个空的bytearray对象 - bytearray(10)创建一个长度为10且内容被0填充的bytearray - bytearray(iterable)会将可迭代对象的内容转换成bytes然后存入对象中 - bytearray(b'Hi!')将已有的二进制数据复制进对象

另外bytearray还提供了fromhex和hex方便将数据以16进制的形式输入输出:

>>> array.hex()
'e6b58be8af95e794a8e4be8b62'
>>> bytearray().fromhex('e6b58be8af95e794a8e4be8b62').decode('utf8')'测试用例b'

4.memoryview

memoryview提供了直接访问对象内存的机制,只要目标对象支持[buffer protocol],例如bytesbytearray

memoryview有个称为“元素”的概念,也就是对象规定的最小的内存单元,比如bytes和bytearray的最小内存单元就是一个byte,具体取决于对象的实现。

len(view)通常等于len(view.tolist()),也就是等于view的“元素”数量。如果view.ndim == 0,那么整个view的内存会被视作一个整体,len会返回1,如果view.ndim == 1那么就正常返回“元素”的个数。view.itemsize会返回单个“元素”的大小。单位是byte。

view.readonly表示当前的memoryview是否是只读的,例如bytes对象的view就是只读的,view.readonly的值为True。是否只读取决于被引用的对象是否可变以及对buffer protocol的实现。

对于使用完毕的memoryview应该尽快调用其release()方法释放资源,而且部分对象在被view引用时会自动进行一些限制,比如bytearray会禁止调整大小,及时释放view是资源可以解除这些限制。

结合示例可以更清晰地了解这些特性:

>>> data = bytearray(b'abcefg')
>>> v = memoryview(data)
>>> v.readonly
False
>>> v[0] = ord(b'z')
>>> data
bytearray(b'zbcefg')
>>> v[1:4] = b'123'
>>> data
bytearray(b'z123fg')
>>> v[2:3] = b'spam'
Traceback (most recent call last):File "<stdin>", line 1, in <module>
ValueError: memoryview assignment: lvalue and rvalue have different structures
>>> v[2:6] = b'spam'
>>> databytearray(b'z1spam')

5.dict-views

准确的说,这不是一种类型,而是一种概念。然而typing里仍然将其视为一种类型,所以也就罗列在此了。

概念:返回自dict.keys(),dict.values()和dict.items()的对象被称作dict-views。

对于views对象,可以使用len,成员检测,它本身也是可迭代对象:

>>> dishes = {'eggs': 2, 'sausage': 1, 'bacon': 1, 'spam': 500}
>>> keys = dishes.keys()
>>> values = dishes.values()>>> # iteration
>>> n = 0
>>> for val in values:
...     n += val
>>> print(n)
504>>> # keys and values are iterated over in the same order (insertion order)
>>> list(keys)
['eggs', 'sausage', 'bacon', 'spam']
>>> list(values)
[2, 1, 1, 500]
#学习中遇到问题没人解答?小编创建了一个Python学习交流群:153708845>>> # view objects are dynamic and reflect dict changes
>>> del dishes['eggs']
>>> del dishes['sausage']
>>> list(keys)
['bacon', 'spam']>>> # set operations
>>> keys & {'eggs', 'bacon', 'salad'}
{'bacon'}
>>> keys ^ {'sausage', 'juice'}{'juice', 'sausage', 'bacon', 'spam'}

从例子中可以看出,views保持着元素的插入顺序(插入顺序的保证从python3.6开始)以及views动态反应了key/value的插入和删除以及修改,因此在某些场景下views对象是相当有用的。

6.The Ellipsis Object (...)

...不是一个类型,不过算是一个内置对象。

它没什么特殊的含义,仅表示省略,通常被用在type hints中:

>>> ...
Ellipsis
>>> from typing import Callable>>> func: Callable[..., None] = lambda x,y:print(x*y)

func是一个没有返回值的函数,参数列表没有做任何限制。

你也可以写成Ellipsis,两者是等价的,不过显然是...这种形式更简单明了。

以上就是这些容易被忽略和遗忘的内置类型,如有错误和疏漏欢迎指出。

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

如若内容造成侵权/违法违规/事实不符,请联系编程大学网进行投诉反馈email:xxxxxxxx@qq.com,一经查实,立即删除!

相关文章

Python-数字取证秘籍(三)

Python 数字取证秘籍(三)原文:zh.annas-archive.org/md5/941c711b36df2129e5f7d215d3712f03 译者:飞龙 协议:CC BY-NC-SA 4.0第六章:阅读电子邮件和获取名称的配方 本章涵盖了以下配方:解析 EML 文件查看 MSG 文件订购外卖盒子里有什么?解析 PST 和 OST 邮箱介绍 一旦计…

一个.NET内置依赖注入的小型强化版

前言 .NET生态中有许多依赖注入容器。在大多数情况下,微软提供的内置容器在易用性和性能方面都非常优秀。外加ASP.NET Core默认使用内置容器,使用很方便。 但是笔者在使用中一直有一个头疼的问题:服务工厂无法提供请求的服务类型相关的信息。这在一般情况下并没有影响,但是…

宝塔面板mysql无法启动问题如何解决

宝塔面板无法启动的问题和解决 如果你的宝塔里面的mysql无法启动了,请先看是不是以下的配置问题 1.是不是你的3306端口被占用了导致mysql无法启动 2.是不是磁盘空间不足导致的无法启动 如果都不是这些问题再继续向下看 常见问题: 1、Mysql安装好后或迁移文件后无法启动 2、My…

小程序上显示富文本

功能:富文本内容格式化、拿到富文本里的所有图片点击放大图片util.tsexport function formatRichText(html: any) { // 富文本内容格式化let arrText = html;//正则匹配不含style="" 或 style= 的img标签var regex1 = new RegExp("(i?)(\<img)(?!(.*?sty…

TPU-MLIR实现技术详细介绍

TPU-MLIR实现技术详细介绍 TPU-MLIR简介 后文假定用户已经处于docker里面的/workspace目录。 编译ONNX模型 以 yolov5s.onnx 为例, 介绍如何编译迁移一个onnx模型至BM1684X TPU平台运行。 该模型来自yolov5的官网: https://github.com/ultralytics/yolov5/releases/download/v6…

PostgreSQL源码学习 win10源码编译安装

源码学习的第一步是源码安装,只有用源码安装才能之后在此基础上阅读、调试、开发系统。 我这里安装的是PostgreSQL 13.2版本,其他版本大同小异,如有出入,以最新版本的文档为准:PostgreSQL最新版文档 一、下载源码 源码下载链接,内含各自版本的源码:PG源码 比如13.2版本的…