Python3.10的一些新特性原理分析


Posted in Python onSeptember 15, 2021

Python 3.10.0a2 版本已经于 2020-11-04 发布,因此我们可以窥见 Python 3.10 的一些新特性。这些新特性很可能会改变未来的 Python 生态系统,使其朝着更明确,更易读的方向发展,同时保持我们熟知和喜欢的易用性。

Python3.10的一些新特性原理分析

 

 

PEP 602

 

1、类型注释的进一步扩展

3.9 版对 Python 中的类型提示和注释进行了大幅度修改和清理,类型提示这似乎是一种持续的趋势,在 3.10 中得到进一步扩展,目的很明显,是为了更好的可读性,无需看代码即可得知变量和函数返回值的类型。

延迟类型注释的执行

类型注释的运行通常被认为是在函数定义时执行,这意味着类型注释以自上而下的方式逐行进行检查。

尽管看起来合乎逻辑,但是这样做有两个问题:

  • 1、引用尚未定义的类型的类型提示(前向引用)将不起作用,必须以字符串形式表示。也就是说:假如 int 是自定义类型,我们需要编写 "int" 而不是编写 int 。
  • 2、这会减慢模块导入的速度,因为此时会执行类型提示。

因此,取而代之的是延迟类型注释,将类型注释将以字符串形式存储在__annotations__中,如果需要这些类型注释可以在运行时通过 typing.get_type_hints() 来解析,也可以通过inspect.signature() 来立即进行解析,这样的好处是可以先执行模块导入,允许前向引用,从而减少初始化时间。

新增类型注释联合操作符

3.10 通过 "|" 作为逻辑或操作符。在注释数据类型时,我们可以使用 | 作为或。例如,我们有一个预期为 int 或 float 的变量,可以写为 int | float ,如下所示:

def f(x: int | float) -> float:
    return x * 3.142
f(1)  # pass
f(1.5)  # pass
f('str')  # linter will show annotation error

也可以使用 typing 模块提供的关键字 Union,比如 Union[int, float]

TypeAlias 注释

回到前向引用问题,避免前向引用的常见解决方案是将它们编写为字符串。

但是,将类型写为字符串会在将这些类型分配给变量时引起问题,因为 Python 会假定我们的字符串文字类型注释只是一个字符串。

在通常使用类型注释的地方使用该类型注释变量将返回错误。例如:

MyType = "ClassName"  # ClassName is our type annotation
def foo() -> MyType:
    ...

在这里,我们试图将其 MyType 用作类型的别名 ,但是, MyType 它将被读取为字符串值,而不是类型别名。只要 ClassName 在代码的后面定义,这就是有效的。当前情况下,这将引发注释错误。

为了解决这个问题,添加了一种显式标识 MyType 为类型别名的方法 :

from typing_extensions import TypeAlias
MyType: TypeAlias = "ClassName"
def foo() -> MyType:
    ...
OR
MyType: TypeAlias = ClassName # if we have defined ClassName already
def foo() -> MyType:
    ...

这里说下,为什么类型很重要,尽管这当然不是一个巨大的变动,但是看到 Python 开发人员加倍努力以增强类型功能,这真是太酷了。Python 的优势在于其易用性和缺乏陡峭的学习曲线。原因之一是不需要在我们的代码中显式定义类型。

增强类型注释看起来似乎违反直觉,但是为开发人员 提供定义类型的选项 可以极大地提高代码库的可读性和可维护性。例如,从 Python transformers 库的源代码中可以看到以下说明:

Even without context, we can read this code and immediately grasp what data we should expect to be fed into these functions, classes, and methods — and exactly which datatypes we should be expecting to return.

In complex code bases (and even simple ones), type annotation can massively improve readability. Simultaneously, not everyone will want (or need) to use them — so an optional, exception-free functionality strikes a perfect balance.

意思是即使没有上下文,我们也可以阅读此代码,并立即掌握应将哪些数据期望输入到这些函数,类和方法中,以及确切地期望返回哪些数据类型。

但在复杂的代码库(甚至简单的代码库)中,类型注释可以大大提高可读性。同时,并不是每个人都希望(或需要)使用它们,因此,这是可选的。这种无异常的功能可以达到完美的平衡。

这些改进表明 Python 对类型注释功能的承诺,基于此,我们最喜欢的库和我们自己写的代码可以大大提示可阅读性,这会对 Python 生态系统产生长期的正面影响。

2、新增的函数及函数参数的变化

除了类型提示功能的扩展外,核心 Python 功能进行了一些更新,如下。

函数 zip() 增加 strict 参数

函数 zip() 增加 strict 参数,如果设置 strict = True,而传输的参数的长度不相等将会抛出异常,如下图所示:

Python3.10的一些新特性原理分析

新的 strict 参数不是盲目地截断不匹配的数据,而是使我们能够控制它的行为,这将使很多开发人员免于遭受麻烦。

新增整数的位计数器 int.bit_count()

此新方法使我们能够计算整数的二进制表示形式中 1 的个数,在某些场景下这个函数非常实用且高效。

Python3.10的一些新特性原理分析

上图中的结果即为整数以二进制位为 1 的个数:

0   = 00000000
1   = 00000001
2   = 00000010
3   = 00000011
10  = 00001010
11  = 00001011
12  = 00001100
100 = 01100100
101 = 01100101
102 = 01100110

字典的视图增加一个属性

字典类型的 3 个方法:dict.items()、dict.keys()、dict.values() 分别返回字典的 3 个视图,现在每个视图都增加来一个属性,叫 mapping,具体用法如下:

Python3.10的一些新特性原理分析

新的属性 mapping 的类型属于 types.MappingProxyType,是围绕原字典的一个属性,在任何视图上访问 mapping 属性,都将返回原字典。

现在就这些了,尽管我们距离 3.10 的开发时间表只有几个月的时间,但已经有很多有趣的更改,Python 的发展仍在继续,似乎还会为语言添加更多有趣的功能。

以上就是Python3.10的一些新特性原理分析的详细内容,更多关于Python3.10新特性的资料请关注三水点靠木其它相关文章!

Python 相关文章推荐
python基础教程之popen函数操作其它程序的输入和输出示例
Feb 10 Python
python使用正则表达式检测密码强度源码分享
Jun 11 Python
进一步理解Python中的函数编程
Apr 13 Python
Django实现组合搜索的方法示例
Jan 23 Python
对Python中list的倒序索引和切片实例讲解
Nov 15 Python
Pycharm+Scrapy安装并且初始化项目的方法
Jan 15 Python
Python实现根据日期获取当天凌晨时间戳的方法示例
Apr 09 Python
pandas 对日期类型数据的处理方法详解
Aug 08 Python
python 安装库几种方法之cmd,anaconda,pycharm详解
Apr 08 Python
Pyinstaller加密打包应用的示例代码
Jun 11 Python
python获取系统内存占用信息的实例方法
Jul 17 Python
在python中读取和写入CSV文件详情
Jun 28 Python
一篇文章带你了解Python和Java的正则表达式对比
Sep 15 #Python
Python编程编写完善的命令行工具
Sep 15 #Python
python可视化之颜色映射详解
python的变量和简单数字类型详解
Sep 15 #Python
深入浅析Django MTV模式
python 进阶学习之python装饰器小结
Sep 04 #Python
自动在Windows中运行Python脚本并定时触发功能实现
Sep 04 #Python
You might like
PHP SEO优化之URL优化方法
2011/04/21 PHP
PHP实现获取FLV文件的时间
2015/02/10 PHP
thinkPHP下ueditor的使用方法详解
2015/12/26 PHP
解决在laravel中auth建立时候遇到的问题
2019/10/15 PHP
jquery中常用的SET和GET
2009/01/13 Javascript
将函数的实际参数转换成数组的方法
2010/01/25 Javascript
javascript setTimeout和setInterval计时的区别详解
2013/06/21 Javascript
使用js实现关闭js弹出层的窗口
2014/02/10 Javascript
JS实现跟随鼠标立体翻转图片的方法
2015/05/04 Javascript
JavaScript的RequireJS库入门指南
2015/07/01 Javascript
Bootstrap组件(一)之菜单
2016/05/11 Javascript
jQuery autoComplete插件两种使用方式及动态改变参数值的方法详解
2016/10/24 Javascript
对Angular.js Controller如何进行单元测试
2016/10/25 Javascript
jQuery实现贪吃蛇小游戏(附源码下载)
2017/03/04 Javascript
用Axios Element实现全局的请求loading的方法
2018/03/15 Javascript
微信小程序上传图片实例
2018/05/28 Javascript
vue 组件中添加样式不生效的解决方法
2018/07/06 Javascript
微信实现自动跳转到用其他浏览器打开指定APP下载
2019/02/15 Javascript
layui默认选中table的CheckBox复选框方法
2019/09/19 Javascript
理解JavaScript中的Proxy 与 Reflection API
2020/09/21 Javascript
node.js如何操作MySQL数据库
2020/10/29 Javascript
vue 计算属性和侦听器的使用小结
2021/01/25 Vue.js
搭建Python的Django框架环境并建立和运行第一个App的教程
2016/07/02 Python
基于Python列表解析(列表推导式)
2018/06/23 Python
Django 对IP访问频率进行限制的例子
2019/08/30 Python
python双向链表原理与实现方法详解
2019/12/03 Python
使用python实现数组、链表、队列、栈的方法
2019/12/20 Python
pytorch 修改预训练model实例
2020/01/18 Python
python使用nibabel和sitk读取保存nii.gz文件实例
2020/07/01 Python
Python常用base64 md5 aes des crc32加密解密方法汇总
2020/11/06 Python
常用的四种CSS透明属性介绍
2014/04/12 HTML / CSS
HTML5 背景的显示区域实现
2020/07/09 HTML / CSS
国际领先的学术出版商:Springer
2017/01/11 全球购物
半年思想汇报
2013/12/30 职场文书
意向协议书
2015/01/27 职场文书
故意杀人案辩护词
2015/05/21 职场文书