浅谈怎么给Python添加类型标注


Posted in Python onJune 08, 2021

Python 添加类型标注

Python 如此简洁,书写者在声明变量时甚至无需考虑类型。

但是简洁与复杂间,是存在一个平衡点的。当我们书写较为复杂的项目时,还是希望可以拥有「静态类型语言」强大的类型检查和智能的提示。

好消息是,并不需要像 TypeScript 那样,引入一个新的编译器来给 JavaScript 做“升级”来进行类型检查, Python 自带的 typing 工具可以在一定程度上把 Python 变成「静态类型语言」;坏消息是, Python 归根结底不是「静态类型语言」,经过我的简单测试,其代码还是「自由松散」的。

给 Python 标注类型

首先和读者声明我们的实验环境。

❯ python --version
Python 3.7.0

本文参考了 Python3.7 的 中文文档

我们声明一个变量,通过如下方式声明类型:

a: int = 1
b: float = .2
print(f'{a}, {b}')  # 1, 0.2

遗憾的是,在 Python 中,a: int = 1 这句话并没什么意义,说的直白点,就是『脱裤子放屁』;再说得好听点, Python 的类型标注放在这里这么用完全没有必要。

首先, a = 1 中解释器会自动把 a 推断为 int 类型,诸如 Pylance 的 Language Server 也会在我们书写时提供 int 的方法补全。

浅谈怎么给Python添加类型标注

此外,就算我们把 a 的类型规定为 int ,然后将 str 赋给 a ,解释器和 Language Server 也完全不会报错。如下。

a: int
a = '1'
print(a)  # 非常迷

做上述类型检查对于现代编译技术而言应该是毫无难度的,但这里就是没有报错、没有警告。这大概与 Python 的设计哲学有关。

我们看看 TypeScript 是如何表现的:

浅谈怎么给Python添加类型标注

TypeScript 把自己当作静态类型语言,要求书写时就确保类型的正确性。

使用 typing

尽管 Python 并不强制要求类型的正确性,并且会自动帮我们做强类型转换,但是我们依旧可以享受类型标注带来的诸多便利。

比如,我们现在要定义一个函数 foo ,函数返回一个列表 dogList ,列表中的元素都是我们自定义的类 Dog 的实例。

如果没有类型标注,我们无法获得智能提示,如下。

浅谈怎么给Python添加类型标注

Python 中从来就不要求 List 对象中的元素都是同一类型,因此,解释器或者 Language Server 也不会「吃力不讨好」般地去把程序运行一遍,然后推断你这个 List 里放的东西是什么类型。

自然,当你从 List 中拿元素时(比如上述的 dogList[0] ),它没法告诉你 List 中你拿的元素是什么类型,也就没办法提示(No suggestion.)。

这与实际业务场景不符,因为我们写代码时,在一个列表中装入的往往都是同一类型。 为了在取元素时获得补全提示,我们可以使用 typing.List + 极简的泛型 。如下。

浅谈怎么给Python添加类型标注

我们规定, foo 返回的元素必是一个 List ,且其中元素类型是 Dog 类型。然后我们的 dogList[0] 也被识别成了 Dog 类型,获得了补全。舒服。

题外话:聪明的 Pylance

其实 Pylance 自己也可以做一些类型推导。比如我们使用生成器生成列表时, Pylance 就会判断这个列表中元素属于什么类型:

浅谈怎么给Python添加类型标注

结语

关于 typing 的用法,还有很多内容可以讨论,我的参考资料主要是:Python3.7 的 typing中文文档 。此外,用 Python 泛型实现函数重载相比静态类型语言似乎十分麻烦(我参考了Python实用宝典的文章(知乎)),如果之后我遇到合适的场景也会成文分享。

到此这篇关于浅谈怎么给Python添加类型标注的文章就介绍到这了,更多相关Python添加类型标注内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
python实现的解析crontab配置文件代码
Jun 30 Python
Python __setattr__、 __getattr__、 __delattr__、__call__用法示例
Mar 06 Python
Python 模板引擎的注入问题分析
Jan 01 Python
Python编程实现二分法和牛顿迭代法求平方根代码
Dec 04 Python
matplotlib.pyplot画图 图片的二进制流的获取方法
May 24 Python
Python中偏函数用法示例
Jun 07 Python
python 读取文本文件的行数据,文件.splitlines()的方法
Jul 12 Python
对Python3中bytes和HexStr之间的转换详解
Dec 04 Python
安装python及pycharm的教程图解
Oct 10 Python
Python hashlib模块加密过程解析
Nov 05 Python
Python 脚本实现淘宝准点秒杀功能
Nov 13 Python
python lambda函数及三个常用的高阶函数
Feb 05 Python
Python如何导出导入所有依赖包详解
Jun 08 #Python
OpenCV-Python实现油画效果的实例
OpenCV-Python实现图像平滑处理操作
OpenCV-Python模板匹配人眼的实例
健身房被搭讪?用python写了个小米计时器助人为乐
解决pycharm安装scrapy DLL load failed:找不到指定的程序的问题
OpenCV-Python实现轮廓拟合
You might like
php公用函数列表[正则]
2007/02/22 PHP
PHP获取网址的顶级域名函数代码
2012/09/24 PHP
Windows下安装PHP单元测试环境PHPUnit图文教程
2014/10/24 PHP
PHP中is_file()函数使用指南
2015/05/08 PHP
PHP获取日期对应星期、一周日期、星期开始与结束日期的方法
2018/06/22 PHP
ThinkPHP5.0框架结合Swoole开发实现WebSocket在线聊天案例详解
2019/04/02 PHP
javascript实现限制上传文件大小
2015/02/06 Javascript
JavaScript中的Math.LOG2E属性使用详解
2015/06/14 Javascript
js实现的倒计时按钮实例
2015/06/24 Javascript
分享一些常用的jQuery动画事件和动画函数
2015/11/27 Javascript
Document.body.scrollTop的值总为零的快速解决办法
2016/06/09 Javascript
原生js实现class的添加和删除简单代码
2016/07/12 Javascript
一种基于浏览器的自动小票机打印实现方案(js版)
2016/07/26 Javascript
Js自动截取字符串长度,添加省略号(……)的实现方法
2017/03/06 Javascript
JavaScript之RegExp_动力节点Java学院整理
2017/06/29 Javascript
微信小程序tabBar底部导航中文注解api详解
2017/08/16 Javascript
详解ionic本地相册、拍照、裁剪、上传(单图完全版)
2017/10/10 Javascript
webpack 打包压缩js和css的方法示例
2018/03/20 Javascript
vue tab滚动到一定高度,固定在顶部,点击tab切换不同的内容操作
2020/07/22 Javascript
Flask SQLAlchemy一对一,一对多的使用方法实践
2013/02/10 Python
python实现上传下载文件功能
2020/11/19 Python
python调用c++传递数组的实例
2019/02/13 Python
python for循环remove同一个list过程解析
2019/08/14 Python
Python任务调度利器之APScheduler详解
2020/04/02 Python
3种适用于Python的疯狂秘密武器及原因解析
2020/04/29 Python
python爬虫中PhantomJS加载页面的实例方法
2020/11/12 Python
Origins悦木之源英国官网:雅诗兰黛集团高端植物护肤品牌
2017/11/06 全球购物
Melissa鞋马来西亚官方网站:MDreams马来西亚
2018/04/05 全球购物
什么是用户模式(User Mode)与内核模式(Kernel Mode) ?
2015/09/07 面试题
Java如何支持I18N?
2016/10/31 面试题
幼儿园春季开学寄语
2014/04/03 职场文书
2014年校长工作总结
2014/12/11 职场文书
孔子观后感
2015/06/08 职场文书
观看禁毒宣传片后的感想
2015/08/11 职场文书
2016年青少年禁毒宣传教育活动总结(学校)
2016/04/05 职场文书
浅谈redis的过期时间设置和过期删除机制
2022/03/18 MySQL