1 python类型注解

        类型注解,即对变量的类型,进行标注或者说明,因为Python是一门动态编译型语言,我们无法在赋值时就定义它的变量类型,所以在Python3.5以上版本新增了类型注解,但仅仅是提示作用,并不能严格控制,这是动态编译型语言的通病,下面来仔细看一下什么是Python的类型注解。

2 函数定义的弊端

        Python是动态语言,变量随时可以被赋值,且赋值为不同的类型,这就与静态语言不同了,变量的类型是在运行期决定的,而静态语言事先就已经定义好了变量的类型了。这是动态语言方便之处,但也是一种弊端,我们无法控制变量的类型,也就无法控制异常的产生。举个栗子

def add(x,y):
return x + y
print(add(1,2))
print(add('s','b'))
print(add(1,'a'))

        当用户传入两个数字时,返回它们的和,但是如果我们传递其他变量呢?比如字符串,因为Python中实现了+号的类型重载,所以说两个字符串的确可以加,但是如果是数字和字符串呢?在Python这种强类型语言中来说,属于非法操作(javascript会隐式转换),而这时,我们就需要对用户传入的数据进行类型判断,不符合本函数的需求,那么就抛个异常,或者提示等等操作,这样就不会引起后续代码在执行期崩溃。如何解决呢?其实主要有两种方式。

  • 函数文档
  • 函数注解

3 函数文档

        在函数中插入说明性文档的方式成为函数文档。

def add(x, y):
"""
This function used to add something
:param x: int object
:param y: int object
:return: int object
"""
return x + y

在函数中,一般是定义语句后的首行使用三对双引号表示。通常存储在函数的__doc__属性中。当用户使用help(函数)时,会被打印在屏幕上。

In [68]: def add(x, y):
...: """
...: This function used to add something
...: :param x: int object
...: :param y: int object
...: :return: int object
...: """
...: return x + y
...: In [69]: help(add)
Help on function add in module __main__: add(x, y)
This function used to add something
:param x: int object
:param y: int object
:return: int object In [70]: print(add.__doc__) This function used to add something
:param x: int object
:param y: int object
:return: int object In [71]:

每次都要使用help来查看函数的说明,的确可以让使用者了解函数的参数以及返回值的类型,但并不是所有人都愿意写doc的,在这个所谓的敏捷开发时代,人们大多会以敏捷开发为借口没时间写,所以这种方法不是很用。

4 函数注解

        Python的函数注解是什么呢?首先来看一下如下代码:

def add(x: int, y: int) -> int:
return x + y
  • 函数的位置形参,和默认值形参后使用冒号分隔,后面用于标识变量期望的类型。
  • 在def语句末尾,使用->符号后 指定用于标识函数执行后的返回值类型。

完成以上定义后,,主要的差别如下图:

当我们在IDE中准备传入非注释类型变量时,IDE会帮我们进行颜色提示,用于表示这里传入的变量有点问题。在编写时我们尚且可以使用这种方式,对我们产生一点'警示',但是当我们写的函数被其他人调用的时候,那么就无法进行'提示'了,这个时候,我们就需要对传入的参数进行类型检查了。

我们来总结一下:

  • 函数注解在Python3.5中引入
  • 对函数的参数、返回值进行类型注解
  • 只对函数的参数做一个辅助的说明,并不对函数参数进行类型检查
  • 提供给第三方工具,做代码分析,发现隐藏的BUG
  • 函数注解的信息,保存在函数的__annotation__属性中。

python3.6 以上还添加了变量的注解:i:int = 10,当然也只是提示的作用。

4.1 annotation属性

在Python中使用__开头的表示符一般被用特殊属性,__annotation__存储的就是函数的签名信息

In [71]: def add(x: int, y: int) -> int:
...: return x + y
...: In [73]: add.__annotations__
Out[73]: {'x': int, 'y': int, 'return': int}

        当我们使用变量注释时,变量名和类型就会存放在函数的__annotations__属性中。那么即然有变量存储,那么我们是不是只需要获取传入的参数,然后和annotations中存储的变量类型进行比较是不是就达到目的了呢?仔细思考一下:

  1. 参数检查势必要在函数执行前,想要在add执行前添加参数判断那么就需要使用装饰器了
  2. __annotations__的值是一个字典,字典是无序的,用户按照位置传进来参数是有序的,如何让它们形成对应关系方便我们检测呢?

下面我们来了解一下inspect模块,它可以帮我们完成这个事情。

5 inspect模块

        官方解释如下:inspect模块提供了几个有用的函数来帮助获取关于活动对象的信息,例如模块、类、方法、函数、回溯、框架对象和代码对象。例如,它可以帮助您检查类的内容、检索方法的源代码、提取并格式化函数的参数列表,或者获取显示详细回溯所需的所有信息。

5.1 常用方法

分类 方法名称 功能
判断 inspect.getmodulename(path) 获取模块名称
inspect.ismodule(object) 是不是个模块
inspect.isclass(object) 是不是个类
inspect.ismethod(object) 是不是一个方法
inspect.isfunction(object) 是不是一个函数
inspect.isgeneratorfunction(object) 是不是一个生成器函数
inspect.isgenerator(object) 是不是一个生成器
inspect.iscoroutinefunction(object) 是不是一个协程函数
获取信息 inspect.getmodulename(path) 获取模块名称
inspect.getsource(object) 获取对象的原码(并不会解析装饰器原码)

5.2 signature类

        首先我们要说的是函数的签名信息:它包含了了函数的函数名、它的参数类型,它所在的类和名称空间及其他信息,签名对象(signature object)表示可调用对象的调用签名信息和它的注解信息,当我们使用signature()时,它会重新返回一个包含可调用对象信息的签名对象。

5.3 parameters属性

        signature类的parameters属性,它里面存放的是函数的参数注解和返回值注解,组成的有序字典,其中参数注解的格式为:参数名称,使用inspect.Parameters类包装的参数注解,这个参数注解很强大,它包含如下常用的方法:

方法名称 含义
empty 等同于inspect._empty表示一个参数没有被类型注释
name 参数的名称
default 参数的默认值,如果一个参数没有默认值,这个属性的值为inspect.empty
annotation 参数的注解类型,如果参数没有定义注解,这个属性的值为inspect.empty
kind 参数的类型

这里的参数类型表示的是inspect内置参数类型(其实就是几个常用的函数参数定义类型而已,只是换个名字而已)

_POSITIONAL_ONLY         = _ParameterKind.POSITIONAL_ONLY       # 位置参数_only
_POSITIONAL_OR_KEYWORD = _ParameterKind.POSITIONAL_OR_KEYWORD # 位置或关键字参数
_VAR_POSITIONAL = _ParameterKind.VAR_POSITIONAL # 可变位置参数
_KEYWORD_ONLY = _ParameterKind.KEYWORD_ONLY # keyword-only参数
_VAR_KEYWORD = _ParameterKind.VAR_KEYWORD # 可变关键字参数

其中POSITIONAL_ONLY,Python中没有被实现。

5.4 获取对象的参数签名

根据上面讲的方法,我们可以通过如下方式,简单的获取参数的签名:

In [11]: import inspect
...:
...: def add(x: int, y: int) -> int:
...: return x + y
...:
...: sig = inspect.signature(add)
...: params = sig.parameters
...: print(params)
OrderedDict([('x', <Parameter "x:int">), ('y', <Parameter "y:int">)]) In [21]: params['x'].annotation
Out[21]: int # 如果没有定义x的参数注解,那么这里就是inspect._empty

通过它的属性,搭配有序字典这个特性,有没有很兴奋?参数有序,传入的实参有序,还能获取参数注解的类型,那么就可以开工进行参数检查了!

6 检查参数

以上面函数为例子,当给add函数传入的x,y时进行参数检查,如果x,y不是int类型,那么返回异常,并退出函数

import inspect
import functools def check(fn):
@functools.wraps(fn) # 等于 wrapper.__annotation__ = fn.__annotation__ 还有其他的属性比如__doc__,__module__等
def wrapper(*args, **kwargs):
sig = inspect.signature(fn) # 获取add函数签名信息
params = sig.parameters # 获取add函数的参数信息
values = list(params.values()) # 由于params是个有序字典,那么values也是有序的,只需根据索一一对应判断即可
for i, k in enumerate(args): # 遍历用户传入的位置参数
if values[i].annotation != inspect._empty: # 如果定义了参数注解,则开始检查
if not isinstance(k, values[i].annotation): # 如果检查不通过,曝出异常
raise('Key Error')
for k,v in kwargs.items():
if params[k].annotation != inspect._empty:
if not isinstance(v,params[k].annotation):
raise('Key Error')
return fn(*args, **kwargs)
return wrapper @check
def add(x: int, y: int) -> int:
return x + y add(4,y=5)

14 - 函数参数检测-inspect模块的更多相关文章

  1. inspect模块详解

    inspect模块主要提供了四种用处: (1).对是否是模块,框架,函数等进行类型检查. (2).获取源码 (3).获取类或函数的参数的信息 (4).解析堆栈 使用inspect模块可以提供自省功能, ...

  2. python inspect 模块 和 types 模块 判断是否是方法,模块,函数等内置特殊属性

    python inspect 模块 和 types 模块 判断是否是方法,模块,函数等内置特殊属性 inspect import inspect def fun(): pass inspect.ism ...

  3. 【集成学习】sklearn中xgboot模块中fit函数参数详解(fit model for train data)

    参数解释,后续补上. # -*- coding: utf-8 -*- """ ############################################## ...

  4. python——inspect模块

    inspect模块常用功能 import inspect # 导入inspect模块 inspect.isfunction(fn) # 检测fn是不是函数 inspect.isgenerator((x ...

  5. inspect模块的使用

    一.介绍 inspect模块主要的四种用处: 1.对是否是模块.框架.函数等进行类型检测 2.获取源码 3.获取类或函数的参数信息 4.解析堆栈 二.使用 只写了2个自己用到的方法,方法太用,http ...

  6. Python常用函数、方法、模块记录

    常用函数: 1.pow():乘方 2.abs():绝对值 3.round():四舍五入 4.int():转换为整数 5.input():键盘输入(会根据用户的输入来做类型的转换) raw_input( ...

  7. JavaScript函数参数与调用

    函数调用: /* 1. 函数调用 */ ,,,); /* 2. 方法调用 */ this.CName = "全局"; var o = { CName:"o类", ...

  8. Python函数参数的五种类型

    之前项目需求,需要通过反射获取函数的参数,python中可以通过函数签名(signature)来实现. 首先需要了解函数参数的类型,Python的参数类型一共有5种:POSITIONAL_OR_KEY ...

  9. Python基础(协程函数、内置函数、递归、模块和包)-day05

    写在前面 上课第五天,打卡: 凭着爱,再回首: 一.协程函数(生成器:yield的表达式形式) 1.yield 的语句形式: yield 1 - 这种方式在 Python基础(函数部分)-day04  ...

随机推荐

  1. FZU2127_养鸡场

    题目的意思为要你求出满足三边范围条件且周长为n的三角形的数目. 其实做法是直接枚举最短边,然后就可以知道第二条边的取值范围,同时根据给定的范围缩小范围. 同时根据第二条边的范围推出第三条边的范围,再次 ...

  2. hdu6415 Rikka with Nash Equilibrium (DP)

    题目链接 Problem Description Nash Equilibrium is an important concept in game theory. Rikka and Yuta are ...

  3. 【bzoj2351】[BeiJing2011]Matrix 二维Hash

    题目描述 给定一个M行N列的01矩阵,以及Q个A行B列的01矩阵,你需要求出这Q个矩阵哪些在原矩阵中出现过.所谓01矩阵,就是矩阵中所有元素不是0就是1. 输入 输入文件的第一行为M.N.A.B,参见 ...

  4. 洛谷 P1972 [SDOI2009]HH的项链

    不是裸题,鉴定完毕. 我是题面 对于这道题,我是离线做的... 树状数组吧,好些点 我们可以很轻易地得到一个很显然的结论,就是关于同一个数,我们只需要记录它不超过当前区间的最后一次出现的位置即可.举例 ...

  5. QT 选择对话框简单示例

    QT 选择对话框简单示例 部分代码: pDialog->addSeparator(); QAction *pmb2 = pDialog->addAction(QString::fromLo ...

  6. 2018九省联考(SHOI2018)

    听说在退役前还能有去外省的机会QAQ D1 9点T1,T2过拍,感觉自己稳得一批,然后边看T3边幻想AK 事实证明我是多么菜多么无知多么傻逼 想T3时太浮躁,最后也没想出来 T2根本没有想过去怀疑自己 ...

  7. hdu1693 Eat the Trees 【插头dp】

    题目链接 hdu1693 题解 插头\(dp\) 特点:范围小,网格图,连通性 轮廓线:已决策点和未决策点的分界线 插头:存在于网格之间,表示着网格建的信息,此题中表示两个网格间是否连边 状态表示:当 ...

  8. BZOJ4408 [Fjoi 2016]神秘数 【主席树】

    题目链接 BZOJ4408 题解 假如我们已经求出一个集合所能凑出连续数的最大区间\([1,max]\),那么此时答案为\(max + 1\) 那么我们此时加入一个数\(x\),假若\(x > ...

  9. CVE-2017-16995 Ubuntu16.04本地提权漏洞复现

    0x01 前言 该漏洞由Google project zero发现.据悉,该漏洞存在于带有 eBPF bpf(2)系统(CONFIG_BPF_SYSCALL)编译支持的Linux内核中,是一个内存任意 ...

  10. 解题:POI 2016 Nim z utrudnieniem

    题面 出现了,神仙题! 了解一点博弈论的话可以很容易转化题面:问$B$有多少种取(diu)石子的方式使得取后剩余石子异或值为零且取出的石子堆数是$d$的倍数 首先有个暴力做法:$dp[i][j][k] ...