Python自省学习

1. 访问对象的属性

class MyClass():
a=''
b=''
def __init__(self):
pass
def write(self):
print self.a,self.b
myClass=MyClass()
print dir(myClass)
print dir(MyClass) print hasattr(MyClass,'a')
print getattr(MyClass,'a')
print setattr(MyClass,'a','')
print getattr(MyClass,'a')

输出:

['__doc__', '__init__', '__module__', 'a', 'b', 'write']
['__doc__', '__init__', '__module__', 'a', 'b', 'write']
True
1
None
11

访问对象属性的方法有,dir,hasattr,getattr,setattr

用setattr可以修改类的熟悉(不是实例),这个比较厉害。

import os
class MyClass():
'''
aaa
''' a=''
b=''
def __init__(self):
pass
def write(self):
print self.a,self.b
myClass=MyClass() print dir(MyClass)
print MyClass.__name__ #对象的名称
print MyClass.write.func_code #对象的代码
print MyClass.__doc__ #对象的文档
print MyClass.__dict__ #对象所有的属性和方法,与dir类似,但是格式是字典
print os.__file__ #对象所在的路径

输出:

['__doc__', '__init__', '__module__', 'a', 'b', 'write']
MyClass
<code object write at 0000000001D71530, file "E:/myDemo/myDemo/��ʡѧϰ/z1.py", line 13> aaa {'a': '', '__module__': '__main__', 'b': '', 'write': <function write at 0x00000000025AEC18>, '__doc__': '\n aaa\n ', '__init__': <function __init__ at 0x00000000025AEBA8>}
C:\Python27\lib\os.pyc

__doc__返回这个对象的文档,对象下面用三个引号括住的为文档

print isinstance(myClass,MyClass)   #检验某实例对象是否是某个类型

总结:

方法或属性 作用
dir() 返回一个对象拥有的方法和属性的列表
__dict__ 返回一个对象拥有的属性和方法的字典,如果对类实例用,只会返回属性,如果对类用,不会返回父类的函数和属性
__name__ 返回对象名称
__doc__ 返回对象的文档
__self__ 返回对象自身
__file__ 返回对象的路径
__module__ 返回对象所在的模块
__func__ 返回对象的代码
__class__ 返回对象的类
isinstace 检验某实例对象是否是某个类型

 2.代码块

•co_argcount: 普通参数的总数,不包括*参数和**参数。

•co_names: 所有的参数名(包括*参数和**参数)和局部变量名的元组。

•co_varnames: 所有的局部变量名的元组。

•co_filename: 源代码所在的文件名。

•co_flags: 这是一个数值,每一个二进制位都包含了特定信息。较关注的是0b100(0x4)和0b1000(0x8),如果co_flags & 0b100 != 0,说明使用了*args参数;如果co_flags & 0b1000 != 0,说明使用了**kwargs参数。另外,如果co_flags & 0b100000(0x20) != 0,则说明这是一个生成器函数(generator function)。

class MyClass():
'''
aaa
''' a=''
b=''
def __init__(self):
pass
def write(self,c,d):
print self.a,self.b
myClass=MyClass() print MyClass.write.func_code.co_argcount #
print MyClass.write.func_code.co_names # ('a', 'b')
print MyClass.write.func_code.co_varnames # ('self', 'c', 'd')
print MyClass.write.func_code.co_filename # E:/myDemo/myDemo/��ʡѧϰ/z2.py

3.栈帧(frame)

•f_back: 调用栈的前一帧。

•f_code: 栈帧对应的code对象。

•f_locals: 用在当前栈帧时与内建函数locals()相同,但你可以先获取其他帧然后使用这个属性获取那个帧的locals()。

•f_globals: 用在当前栈帧时与内建函数globals()相同,但你可以先获取其他帧……。

import inspect

def add(x, y=1):
f = inspect.currentframe()
print f.f_locals # {'y': 1, 'x': 2, 'f': <frame object at 0x00000000026C19B8>}
print f.f_globals #返回全局变量 {'__builtins__': <module '__builtin__' (built-in)>, '__file__':
'E:/myDemo/myDemo/\xd7\xd4\xca\xa1\xd1\xa7\xcf\xb0/z2.py', 'inspect': <module 'inspect' from 'C:\Python27\lib\inspect.pyc'>,
'__author__': 'kevinlu1010@qq.com', 'add': <function add at 0x00000000026C2518>, '__name__': '__main__', '__package__': None,
'os': <module 'os' from 'C:\Python27\lib\os.pyc'>, '__doc__': None} print f.f_code # <code object add at 0000000001DD6E30, file "E:/myDemo/myDemo/��ʡѧϰ/z2.py", line 6>
print f.f_back # <frame object at 0x00000000025D01C8>
return x+y
add(2)

4.追踪

•tb_next: 追踪的下一个追踪对象。

•tb_frame: 当前追踪对应的栈帧。

•tb_lineno: 当前追踪的行号。

import sys
def div(x, y):
try:
return x/y
except:
tb = sys.exc_info()[2] # return (exc_type, exc_value, traceback)
print tb
print tb.tb_lineno # "return x/y" 的行号 6
print tb.tb_frame.f_locals #返回跳出程序时的栈帧 {'y': 0, 'x': 1, 'tb': <traceback object at 0x000000000271FE88>}
div(1, 0)

同时可以把错误的信息记录起来,后面可以print或者输出到日志上

try:
a=1/0
except:
s=traceback.format_exc()
print s

参考:http://developer.51cto.com/art/201101/242703.htm

上一篇:js循环读取json数据,将读取到的数据用js写成表格


下一篇:Authentication(Spring Security 认证笔记)