2014-10-28 29 views
6

Python'da işlevlerin ve işlevlerin başvurduğu genel öğelerin listesini almak için fname.__code__.co_names'u kullanabilirsiniz. Ben fname.__code__.co_varnames yaparsam, bu iç fonksiyonları içerir, inanıyorum.python - bir işlevin tüm iç işlevlerini listeler?

Temel olarak inner.__code__.co_names yapmak için bir yolu var mı? 'inner' gibi görünen bir dize ile başlayarak, co_varnames tarafından döndürülür?

+0

Bu tek düzeyde bir şey mi yoksa yuvalanmış işlevler mi istiyorsunuz? –

cevap

3

bende yok' de iç içe işlevlerini desteklemek için her kod karşılaşabileceğiniz nesnenin üzerine yinelemeli döngü gerekir Kod nesnesini inceleyebilirsiniz çünkü iç işlevler tembeldir ve kod nesneleri yalnızca c tam zamanında ıslatıldı. Muhtemelen bakmak istediğiniz şey ast modülüdür. İşte size kısa bir örnek:

import ast, inspect 

# this is the test scenario 
def function1(): 
    f1_var1 = 42 
    def function2(): 
     f2_var1 = 42 
     f2_var2 = 42 
     def function3(): 
      f3_var1 = 42 

# derive source code for top-level function 
src = inspect.getsource(function1) 

# derive abstract syntax tree rooted at top-level function 
node = ast.parse(src) 

# next, ast's walk method takes all the difficulty out of tree-traversal for us 
for x in ast.walk(node): 
    # functions have names whereas variables have ids, 
    # nested-classes may all use different terminology 
    # you'll have to look at the various node-types to 
    # get this part exactly right 
    name_or_id = getattr(x,'name', getattr(x,'id',None)) 
    if name_or_id: 
     print name_or_id 

sonuçlar şunlardır: fonksiyonu1, İşlevsiz2, f1_var1, 3 işlevi, f2_var1, f2_var2, f3_var1. Zorunlu feragatname: Muhtemelen bu tür bir şey yapmak için iyi bir neden yoktur .. ama eğlenin :)

Oh ve sadece iç fonksiyonların isimlerini istiyorsan?

print dict([[x.name,x] for x in ast.walk(ast.parse(inspect.getsource(some_function))) if type(x).__name__=='FunctionDef']) 
+0

İyi düşünce. Bu bir "pyc" den çalışıyorsa başarısız olur, ama sanırım bu sıra dışı. – goncalopp

+0

Teşekkürler, bu benim ne aradığım gibi görünüyor. Bunu yapmamın sebebi, fonksiyon izlerini yapma girişimi yapıyorum ve içsel işlevleri buna dahil etmek istiyorum. – user3475234

3

Python 3.4+ sürümünde dis.get_instructions kullanarak adlarını alabilirsiniz.

import dis 
import types 

def get_names(f): 
    ins = dis.get_instructions(f) 
    for x in ins: 
     try: 
      if x.opcode == 100 and '<locals>' in next(ins).argval\ 
               and next(ins).opcode == 132: 
       yield next(ins).argrepr 
       yield from get_names(x.argval) 
     except Exception: 
      pass 

Demo:

def func(): 
    x = 1 
    y = 2 
    print ('foo') 
    class A: 
     def method(self): 
      pass 
    def f1(): 
     z = 3 
     print ('bar') 
     def f2(): 
      a = 4 
      def f3(): 
       b = [1, 2, 3] 
    def f4(): 
     pass 

print(list(get_names(func))) 

Çıkışlar:

['f1', 'f2', 'f3', 'f4'] 
İlgili konular