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学习之asyncore模块用法实例教程
Sep 29 Python
python+Django+apache的配置方法详解
Jun 01 Python
正确理解python中的关键字“with”与上下文管理器
Apr 21 Python
人工智能最火编程语言 Python大战Java!
Nov 13 Python
用python 批量更改图像尺寸到统一大小的方法
Mar 31 Python
Python简单实现阿拉伯数字和罗马数字的互相转换功能示例
Apr 17 Python
Python分支语句与循环语句应用实例分析
May 07 Python
python版DDOS攻击脚本
Jun 12 Python
Django框架静态文件使用/中间件/禁用ip功能实例详解
Jul 22 Python
python自动循环定时开关机(非重启)测试
Aug 26 Python
django下创建多个app并设置urls方法
Aug 02 Python
python超详细实现完整学生成绩管理系统
Mar 17 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获取服务器时间的实现代码
2013/06/07 PHP
关于php循环跳出的问题
2013/07/01 PHP
PHP利用str_replace防注入的方法
2013/11/10 PHP
php伪静态验证码不显示的解决方案
2019/09/26 PHP
Laravel timestamps 设置为unix时间戳的方法
2019/10/11 PHP
javaScript 简单验证代码(用户名,密码,邮箱)
2009/09/28 Javascript
Javascript倒计时页面跳转实例小结
2013/09/11 Javascript
JQuery中$(document)是什么意思有什么作用
2014/07/21 Javascript
js点击按钮实现带遮罩层的弹出视频效果
2015/12/19 Javascript
JavaScript小技巧整理
2015/12/30 Javascript
vue增删改查的简单操作
2017/07/15 Javascript
vue webuploader 文件上传组件开发
2017/09/23 Javascript
vue2 router 动态传参,多个参数的实例
2017/11/10 Javascript
通过webpack引入第三方库的方法
2018/07/20 Javascript
微信小程序批量监听输入框对按钮样式进行控制的实现代码
2019/10/12 Javascript
微信小程序实现搜索功能
2020/03/10 Javascript
[41:41]TFT vs Secret Supermajor小组赛C组 BO3 第一场 6.3
2018/06/04 DOTA
python实现简单socket程序在两台电脑之间传输消息的方法
2015/03/13 Python
介绍Python的Django框架中的QuerySets
2015/04/20 Python
Python 常用的安装Module方式汇总
2017/05/06 Python
python 随机数使用方法,推导以及字符串,双色球小程序实例
2017/09/12 Python
基于python 处理中文路径的终极解决方法
2018/04/12 Python
对pandas replace函数的使用方法小结
2018/05/18 Python
使用python将图片格式转换为ico格式的示例
2018/10/22 Python
python+pyqt5实现KFC点餐收银系统
2019/01/24 Python
Python MySQL 日期时间格式化作为参数的操作
2020/03/02 Python
pycharm下配置pyqt5的教程(anaconda虚拟环境下+tensorflow)
2020/03/25 Python
500行python代码实现飞机大战
2020/04/24 Python
jenkins+python自动化测试持续集成教程
2020/05/12 Python
如何更换python默认编辑器的背景色
2020/08/10 Python
Python中lru_cache的使用和实现详解
2021/01/25 Python
2014年综合治理工作总结
2014/11/20 职场文书
答谢词范文
2015/01/05 职场文书
2015暑假社会调查报告
2015/07/13 职场文书
Python基础知识学习之类的继承
2021/05/31 Python
Redis读写分离搭建的完整步骤
2021/09/14 Redis