Ram*_*hum 12 python finally return-value
我在try子句中有一个return语句:
def f():
try:
return whatever()
finally:
pass # How do I get what `whatever()` returned in here?
Run Code Online (Sandbox Code Playgroud)
是否有可能在finally子句中获得返回值?
这更像是一个理论问题,所以我不是在寻找一种解决方法,比如将它保存到变量中.
你是否必须在退货声明后"进入"?
语句之前允许的更改,sys.settrace()就是您所需要的.
返回后才:
我认为,在无堆栈Python中,你应该能够做到这一点."线程"可以在无堆栈中进行腌制,并且即将返回的值(也就是值堆栈的顶部)应该存在.
在CPython中,我无法找到价值堆栈顶部的方式.
当然,通过ctypes或c级扩展可以实现一切,这是一个快速演示:
"""
valuestack.py: Demo reading values from Python value stack
Offsets are derived for CPython-2.7.2, 64-bit, production build
"""
import ctypes, inspect, random
def id2obj(i):
"""convert CPython `id` back to object ref, by temporary pointer swap"""
tmp = None,
try:
ctypes.cast(id(tmp), ctypes.POINTER(ctypes.c_ulong))[3] = i
return tmp[0]
finally:
ctypes.cast(id(tmp), ctypes.POINTER(ctypes.c_ulong))[3] = id(None)
def introspect():
"""pointer on top of value stack is id of the object about to be returned
FIXME adjust for sum(vars, locals) in introspected function
"""
fr = inspect.stack()[1][0]
print "caught", id2obj(ctypes.cast(id(fr), ctypes.POINTER(ctypes.c_ulong))[47])
def value():
tmp = random.random()
print "return", tmp
return tmp
def foo():
try:
return value()
finally:
introspect()
if __name__ == "__main__":
foo()
Run Code Online (Sandbox Code Playgroud)
与osx一起提供的64位模式下的Python-2.7.2:
air:~ dima$ python valuestack.py
return 0.959725159294
caught 0.959725159294
Run Code Online (Sandbox Code Playgroud)