# 让装饰器对函数的参数进行强制执行检查 from inspect import signature from functools import wraps #定义装饰器@typeassert def typeassert(*ty_args, **ty_kwargs): def decorate(func): # If in optimized mode, disable type checking if not __debug__: return func # Map function argument names to supplied types sig = signature(func) bound_types = sig.bind_partial(*ty_args, **ty_kwargs).arguments @wraps(func) def wrapper(*args, **kwargs): bound_values = sig.bind(*args, **kwargs) # Enforce type assertions across supplied arguments for name, value in bound_values.arguments.items(): if name in bound_types: if not isinstance(value, bound_types[name]): raise TypeError( 'Argument {} must be {}'.format(name, bound_types[name]) ) return func(*args, **kwargs) return wrapper return decorate @typeassert(int, z=int) def spam(x, y, z=42): print(x, y, z) print(spam(1, 2, 3)) # 1 2 3 print(spam(1, 'hello', 3)) # 1 hello 3 print(spam(1, 'hello', 'world')) # ypeError: Argument z must be <class 'int'>