基于装饰器和inspect模块对函数参数类型进行检查。
from functools import wraps from inspect import signature
def assert_type(*type_args, **type_kwargs):
def is_type(value, tpe):
return isinstance(value, tpe)
def dec(fn):
sig = signature(fn)
bound_types = sig.bind_partial(*type_args, **type_kwargs).arguments
@wraps(fn)
def wrapper(*args, **kwargs):
bound_values = sig.bind(*args, **kwargs)
for name, value in bound_values.arguments.items():
if name in bound_types:
if not is_type(value, bound_types[name]):
raise TypeError('Argument {} must be {}'.format(name, bound_types[name]))
return fn(*args, **kwargs)
return wrapper
return dec
上面的代码中,使用inspect
中的signature
方法获取了fn
的Signature签名
,然后使用bind_partial
方法创建了(*type_args, **type_kwargs)
到func
参数的映射(也就是一个字典)。
# 使用
@assert_type(a=int, b=int) def add(a, b=1): return a+b
向量化版本:
def vectorized_assert_type(*type_args, **type_kwargs): @np.vectorize def is_type(value, tpe): return isinstance(value, tpe) def dec(fn): sig = signature(fn) bound_types = sig.bind_partial(*type_args, **type_kwargs).arguments @wraps(fn) def wrapper(*args, **kwargs): bound_values = sig.bind(*args, **kwargs) for name, value in bound_values.arguments.items(): if name in bound_types: if not is_type(value, bound_types[name]).all(): raise TypeError('Argument {} must be {}'.format(name, bound_types[name])) return fn(*args, **kwargs) return wrapper return dec