2014-10-28 13 views
6

In python è possibile eseguire fname.__code__.co_names per recuperare un elenco di funzioni e cose globali a cui fa riferimento una funzione. Se faccio fname.__code__.co_varnames, questo include le funzioni interne, credo.python - elenca tutte le funzioni interne di una funzione?

C'è un modo per fare essenzialmente inner.__code__.co_names? iniziando con una stringa che assomiglia a 'inner', come viene restituita da co_varnames?

+0

Si tratta di una cosa di un livello o di funzioni nidificate? –

risposta

3

I don' Penso che sia possibile ispezionare l'oggetto codice perché le funzioni interne sono pigre e i loro oggetti codice sono solo c battuto appena in tempo. Quello che probabilmente vorrai guardare è invece il modulo ast. Ecco un rapido esempio:

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 

I risultati sono: funzione1, function2, f1_var1, funzione3, f2_var1, f2_var2, f3_var1. Disclaimer obbligatorio: probabilmente non c'è una buona ragione per fare questo tipo di cose .. ma divertiti :)

Oh e se vuoi solo i nomi delle funzioni interne?

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

Buon pensiero. Questo fallirebbe se si eseguisse da un 'pyc', ma immagino che sia insolito. – goncalopp

+0

Grazie, sembra proprio quello che avrei cercato. Il motivo per cui lo sto facendo è perché sto facendo un tentativo di fare tracce di funzioni, e voglio includere funzioni interne in questo. – user3475234

3

In Python 3.4+ è possibile ottenere i nomi utilizzando dis.get_instructions. Per supportare funzioni nidificate così è necessario ricorsivamente un ciclo su ogni oggetto di codice si incontrano:

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))) 

Uscite:

['f1', 'f2', 'f3', 'f4'] 
Problemi correlati