Python自省学习

1. 访问对象的属性

class MyClass():
    a='1'
    b='2'
    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','11')
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='1'
    b='2'
    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': '1', '__module__': '__main__', 'b': '2', '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='1'
    b='2'
    def __init__(self):
        pass
    def write(self,c,d):
        print self.a,self.b
myClass=MyClass()

print MyClass.write.func_code.co_argcount  # 3
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

posted @ 2014-07-16 11:11  Xjng  阅读(774)  评论(1编辑  收藏  举报