容易被忽略的Python内置类型


Posted in Python onSeptember 03, 2020

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

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

(注意:本文基于python3,不会包含任何python2相关内容)

frozenset

不可变集合(frozenset)与普通的set一样,只不过它的元素是不可变的,因此诸如`add`,`remove`,`update`等可以添加/删除/改变集合内元素的方法是不存在的,换句话说一旦frozenset建立后你将不再可能更改集合内的元素。其他的方法与set一致: ```python >>> 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}) ```

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既不返回迭代器,也不返回其他可迭代对象,而是返回的自己。

bytearray

`bytearray`一般情况下并不常见,它主要为了可以实现原地修改bytes对象而出现,因为bytes和str一样是不可变对象,例如这样是非法的: ```python >>> 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`时就可以进行原地修改了: ```python >>> array = bytearray(b) >>> array[-1] = 98 >>> array.decode('utf8') 测试用例b ``` `bytearray`对象没有字面常量,因此只能通过构造函数创建,它有着和bytes一样的方法,只是可变以及多了一些序列对象的特性。如果要创建一个`bytearray`可以有如下的几种方法: - `bytearray()`返回一个空的`bytearray`对象 - `bytearray(10)`创建一个长度为10且内容被0填充的`bytearray` - `bytearray(iterable)`会将可迭代对象的内容转换成bytes然后存入对象中 - `bytearray(b'Hi!')`将已有的二进制数据复制进对象

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

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

memoryview

`memoryview`提供了直接访问对象内存的机制,只要目标对象支持[buffer protocol](https://docs.python.org/3/c-api/buffer.html#bufferobjects),例如`bytes`和`bytearray`。
memoryview有个称为“元素”的概念,也就是对象规定的最小的内存单元,比如bytesbytearray的最小内存单元就是一个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'
>>> data
bytearray(b'z1spam')

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]

>>> # 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对象是相当有用的。

The Ellipsis Object (...)

`...`不是一个类型,不过算是一个内置对象。
它没什么特殊的含义,仅表示省略,通常被用在type hints中:

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

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

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

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

参考:

https://docs.python.org/3/library/stdtypes.html

https://docs.python.org/3/c-api/buffer.html#bufferobjects

以上就是容易被忽略的Python内置类型的详细内容,更多关于Python内置类型的资料请关注三水点靠木其它相关文章!

Python 相关文章推荐
python使用mailbox打印电子邮件的方法
Apr 30 Python
Python实现包含min函数的栈
Apr 29 Python
Python实现螺旋矩阵的填充算法示例
Dec 28 Python
Python常见字典内建函数用法示例
May 14 Python
Python 隐藏输入密码时屏幕回显的实例
Feb 19 Python
python中yield的用法详解——最简单,最清晰的解释
Apr 04 Python
python  logging日志打印过程解析
Oct 22 Python
python kafka 多线程消费者&amp;手动提交实例
Dec 21 Python
如何在 Django 模板中输出 &quot;{{&quot;
Jan 24 Python
Python list和str互转的实现示例
Nov 16 Python
Python爬虫之Selenium实现关闭浏览器
Dec 04 Python
Python Matplotlib绘制两个Y轴图像
Apr 13 Python
python类共享变量操作
Sep 03 #Python
Python pip install之SSL异常处理操作
Sep 03 #Python
详解python tkinter 图片插入问题
Sep 03 #Python
解决PyCharm IDE环境下,执行unittest不生成测试报告的问题
Sep 03 #Python
PyTorch中Tensor的数据类型和运算的使用
Sep 03 #Python
python开发入门——set的使用
Sep 03 #Python
使用anaconda安装pytorch的实现步骤
Sep 03 #Python
You might like
收音机的保养
2021/03/01 无线电
Uchome1.2 1.5 代码学习 common.php
2009/04/24 PHP
PHP文件生成的图片无法使用CDN缓存的解决方法
2015/06/20 PHP
php修改数组键名的方法示例
2017/04/15 PHP
PHPExcel中文帮助手册|PHPExcel使用方法(分享)
2017/06/09 PHP
PHP实用小技巧之调用录像的方法
2019/12/05 PHP
深入认识javascript中的eval函数
2009/11/02 Javascript
浅析jquery的js图表组件highcharts
2014/03/06 Javascript
浅谈javascript的Array.prototype.slice.call
2015/08/31 Javascript
JavaScript性能优化之小知识总结
2015/11/20 Javascript
详解基于Node.js的微信JS-SDK后端接口实现代码
2017/07/15 Javascript
Easyui和zTree两种方式分别实现树形下拉框
2017/08/04 Javascript
解决vue-cli项目打包出现空白页和路径错误的问题
2018/09/04 Javascript
vue 弹框产生的滚动穿透问题的解决
2018/09/21 Javascript
layui 富文本图片上传接口与普通按钮 文件上传接口的例子
2019/09/23 Javascript
js表达式与运算符简单操作示例
2020/02/15 Javascript
JavaScript实现打砖块游戏
2020/02/25 Javascript
javascript设计模式 ? 工厂模式原理与应用实例分析
2020/04/09 Javascript
[52:20]VP vs VG Supermajor小组赛 B组胜者组决赛 BO3 第一场 6.2
2018/06/03 DOTA
探究Python多进程编程下线程之间变量的共享问题
2015/05/05 Python
python实现机械分词之逆向最大匹配算法代码示例
2017/12/13 Python
opencv python统计及绘制直方图的方法
2019/01/21 Python
Python实现动态给类和对象添加属性和方法操作示例
2020/02/29 Python
美国伊甸园兄弟种子公司:Eden Brothers
2018/07/01 全球购物
英国泽西岛植物:Jersey Plants Direct
2019/08/07 全球购物
对于没有初始化的变量的初始值可以作怎样的假定
2014/10/12 面试题
中医药大学市场营销专业自荐信
2013/09/29 职场文书
大学生职业规划范文:象牙塔生活的四年计划
2014/01/14 职场文书
校运会入场式解说词
2014/02/10 职场文书
学习雷锋活动总结
2014/04/29 职场文书
2015年七一建党节演讲稿
2015/03/19 职场文书
2015年七一建党节活动方案
2015/05/05 职场文书
焦裕禄观后感
2015/06/03 职场文书
跑吧孩子观后感
2015/06/10 职场文书
校园安全学习心得体会
2016/01/18 职场文书
MySQL连表查询分组去重的实现示例
2021/07/01 MySQL