程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 更多編程語言 >> 更多關於編程 >> Python裝飾器使用實例:驗證參數合法性

Python裝飾器使用實例:驗證參數合法性

編輯:更多關於編程

              這篇文章主要介紹了Python裝飾器使用實例:驗證參數合法性,本文直接給出代碼實例,代碼中包含詳細注釋,需要的朋友可以參考下

       

              python是不帶靜態檢查的動態語言,有時候需要在調用函數時保證參數合法。檢查參數合法性是一個顯著的切面場景,各個函數都可能有這個需求。但另一方面,參數合法性是不是應該由調用方來保證比較好也是一個需要結合實際才能回答的問題,總之雙方約定好,不要都不檢查或者都檢查就可以了。下面這個模塊用於在函數上使用裝飾器進行參數的合法性驗證。

              你可以直接執行這個模塊進行測試,如果完全沒有輸出則表示通過。你也可以找到幾個以_test開頭的函數,所有的測試用例都包含在這幾個函數中。使用方法參見模塊文檔和測試用例。

    ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 # -*- coding: UTF-8 -*-   ''' @summary: 驗證器 該模塊提供了一個裝飾器用於驗證參數是否合法,使用方法為:   from validator import validParam, nullOk, multiType   @validParam(i=int) def foo(i): return i+1   編寫驗證器:   1. 僅驗證類型: @validParam(type, ...) 例如: 檢查第一個位置的參數是否為int類型: @validParam(int) 檢查名為x的參數是否為int類型: @validParam(x=int)   驗證多個參數: @validParam(int, int) 指定參數名驗證: @validParam(int, s=str)   針對*和**參數編寫的驗證器將驗證這些參數實際包含的每個元素: @validParam(varargs=int) def foo(*varargs): pass   @validParam(kws=int) def foo7(s, **kws): pass   2. 帶有條件的驗證: @validParam((type, condition), ...) 其中,condition是一個表達式字符串,使用x引用待驗證的對象; 根據bool(表達式的值)判斷是否通過驗證,若計算表達式時拋出異常,視為失敗。 例如: 驗證一個10到20之間的整數: @validParam(i=(int, '10<x<20')) 驗證一個長度小於20的字符串: @validParam(s=(str, 'len(x)<20')) 驗證一個年齡小於20的學生: @validParam(stu=(Student, 'x.age<20'))   另外,如果類型是字符串,condition還可以使用斜槓開頭和結尾表示正則表達式匹配。 驗證一個由數字組成的字符串: @validParam(s=(str, '/^d*$/'))   3. 以上驗證方式默認為當值是None時驗證失敗。如果None是合法的參數,可以使用nullOk()。 nullOk()接受一個驗證條件作為參數。 例如: @validParam(i=nullOk(int)) @validParam(i=nullOk((int, '10<x<20'))) 也可以簡寫為: @validParam(i=nullOk(int, '10<x<20'))   4. 如果參數有多個合法的類型,可以使用multiType()。 multiType()可接受多個參數,每個參數都是一個驗證條件。 例如: @validParam(s=multiType(int, str)) @validParam(s=multiType((int, 'x>20'), nullOk(str, '/^d+$/')))   5. 如果有更復雜的驗證需求,還可以編寫一個函數作為驗證函數傳入。 這個函數接收待驗證的對象作為參數,根據bool(返回值)判斷是否通過驗證,拋出異常視為失敗。 例如: def validFunction(x): return isinstance(x, int) and x>0 @validParam(i=validFunction) def foo(i): pass   這個驗證函數等價於: @validParam(i=(int, 'x>0')) def foo(i): pass     @author: HUXI @since: 2011-3-22 @change: '''   import inspect import re   class ValidateException(Exception): pass     def validParam(*varargs, **keywords): '''驗證參數的裝飾器。'''   varargs = map(_toStardardCondition, varargs) keywords = dict((k, _toStardardCondition(keywords[k])) for k in keywords)   def generator(func): args, varargname, kwname = inspect.getargspec(func)[:3] dctValidator = _getcallargs(args, varargname, kwname, varargs, keywords)   def wrapper(*callvarargs, **callkeywords): dctCallArgs = _getcallargs(args, varargname, kwname, callvarargs, callkeywords)   k, item = None, None try: for k in dctValidator: if k == varargname: for item in dctCallArgs[k]: assert dctValidator[k](item) elif k == kwname: for item in dctCallArgs[k].values(): assert dctValidator[k](item) else: item = dctCallArgs[k] assert dctValidator[k](item) except: raise ValidateException, ('%s() parameter validation fails, param: %s, value: %s(%s)' % (func.func_name, k, item, item.__class__.__name__))   return func(*callvarargs, **callkeywords)   wrapper = _wrapps(wrapper, func) return wrapper   return generator     def _toStardardCondition(condition): '''將各種格式的檢查條件轉換為檢查函數'''   if inspect.isclass(condition): return lambda x: isinstance(x, condition)   if isinstance(condition, (tuple, list)): cls, condition = condition[:2] if condition is None: return _toStardardCondition(cls)   if cls in (str, unicode) and condition[0] == condition[-1] == '/': return lambda x: (isinstance(x, cls) and re.match(condition[1:-1], x) is not None)   return lambda x: isinstance(x, cls) and eval(condition)   return condition     def nullOk(cls, condition=None): '''這個函數指定的檢查條件可以接受None值'''   return lambda x: x is None or _toStardardCondition((cls, condition))(x)     def multiType(*conditions): '''這個函數指定的檢查條件只需要有一個通過'''   lstValidator = map(_toStardardCondition, conditions) def validate(x): for v in lstValidator: if v(x): return True return validate     def _getcallargs(args, varargname, kwname, varargs, keywords): '''獲取調用時的各參數名-值的字典'''   dctArgs = {} varargs = tuple(varargs) keywords = dict(keywords)   argcount = len(args) varcount = len(varargs) callvarargs = None   if argcount <= varcount: for n, argname in enumerate(args): dctArgs[argname] = varargs[n]   callvarargs = varargs[-(varcount-argcount):]   else: for n, var in enumerate(varargs): dctArgs[args[n]] = var   for argname in args[-(argcount-varcount):]: if argname in keywords: dctArgs[argname] = keywords.pop(argname)   callvarargs = ()   if varargname is not None: dctArgs[varargname] = callvarargs   if kwname is not None: dctArgs[kwname] = keywords   dctArgs.update(keywords) return dctArgs     def _wrapps(wrapper, wrapped): '''復制元數據'''   for attr in ('__module__', '__name__', '__doc__'): setattr(wrapper, attr, getattr(wrapped, attr)) for attr in ('__dict__',): getattr(wrapper, attr).update(getattr(wrapped, attr, {}))   return wrapper     #=============================================================================== # 測試 #===============================================================================     def _unittest(func, *cases): for case in cases: _functest(func, *case)     def _functest(func, isCkPass, *args, **kws): if isCkPass: func(*args, **kws) else: try: func(*args, **kws) assert False except ValidateException: pass   def _test1_simple(): #檢查第一個位置的參數是否為int類型: @validParam(int) def foo1(i): pass _unittest(foo1, (True, 1), (False, 's'), (False, None))   #檢查名為x的參數是否為int類型: @validParam(x=int) def foo2(s, x): pass _unittest(foo2, (True, 1, 2), (False, 's', 's'))   #驗證多個參數: @validParam(int, int) def foo3(s, x): pass _unittest(foo3, (True, 1, 2), (False, 's', 2))   #指定參數名驗證: @validParam(int, s=str) def foo4(i, s): pass _unittest(foo4, (True, 1, 'a'), (False, 's', 1))   #針對*和**參數編寫的驗證器將驗證這些參數包含的每個元素: @validParam(varargs=int) def foo5(*varargs): pass _unittest(foo5, (True, 1, 2, 3, 4, 5), (False, 'a', 1))   @validParam(kws=int) def foo6(**kws): pass _functest(foo6, True, a=1, b=2) _functest(foo6, False, a='a', b=2)   @validParam(kws=int) def foo7(s, **kws): pass _functest(foo7, True, s='a', a=1, b=2)     def _test2_condition(): #驗證一個10到20之間的整數: @validParam(i=(int, '10<x<20')) def foo1(x, i): pass _unittest(foo1, (True, 1, 11), (False, 1, 'a'), (False, 1, 1))   #驗證一個長度小於20的字符串: @validParam(s=(str, 'len(x)<20')) def foo2(a, s): pass _unittest(foo2, (True, 1, 'a'), (False, 1, 1), (False, 1, 'a'*20))   #驗證一個年齡小於20的學生: class Student(object): def __init__(self, age): self.age=age   @validParam(stu=(Student, 'x.age<20')) def foo3(stu): pass _unittest(foo3, (True, Student(18)), (False, 1), (False, Student(20)))   #驗證一個由數字組成的字符串: @validParam(s=(str, r'/^d*$/')) def foo4(s): pass _unittest(foo4, (True, '1234'), (False, 1), (False, 'a1234'))     def _test3_nullok(): @validParam(i=nullOk(int)) def foo1(i): pass _unittest(foo1, (True, 1), (False, 'a'), (True, None))   @validParam(i=nullOk(int, '10<x<20')) def foo2(i): pass _unittest(foo2, (True, 11), (False, 'a'), (True, None), (False, 1))     def _test4_multitype(): @validParam(s=multiType(int, str)) def foo1(s): pass _unittest(foo1, (True, 1), (True, 'a'), (False, None), (False, 1.1))   @validParam(s=multiType((int, 'x>20'), nullOk(str, '/^d+$/'))) def foo2(s): pass _unittest(foo2, (False, 1), (False, 'a'), (True, None), (False, 1.1), (True, 21), (True, '21'))   def _main(): d = globals() from types import FunctionType print for f in d: if f.startswith('_test'): f = d[f] if isinstance(f, FunctionType): f()   if __name__ == '__main__': _main()
    1. 上一頁:
    2. 下一頁:
    Copyright © 程式師世界 All Rights Reserved