I have the following code inside a function:
stored_blocks = {}
def replace_blocks(m):
block = m.group(0)
block_hash = sha1(block)
stored_blocks[block_hash] = block
return '{{{%s}}}' % block_hash
num_converted = 0
def convert_variables(m):
name = m.group(1)
num_converted += 1
return '' % name
fixed = MATCH_DECLARE_NEW.sub('', template)
fixed = MATCH_PYTHON_BLOCK.sub(replace_blocks, fixed)
fixed = MATCH_FORMAT.sub(convert_variables, fixed)
Adding elements to stored_blocks works fine, but I cannot increase num_converted in the second subfunction:
UnboundLocalError: local variable 'num_converted' referenced before assignment
I could use global but global variables are ugly and I really don't need that variable to be global at all.
So I'm curious how I can write to a variable in the parent function's scope.
nonlocal num_converted would probably do the job, but I need a solution that works with Python 2.x.
解决方案
Problem: This is because Python's scoping rules are demented. The presence of the += assignment operator marks the target, num_converted, as local to the enclosing function's scope, and there is no sound way in Python 2.x to access just one scoping level out from there. Only the global keyword can lift variable references out of the current scope, and it takes you straight to the top.
Fix: Turn num_converted into a single-element array.
num_converted = [0]
def convert_variables(m):
name = m.group(1)
num_converted[0] += 1
return '' % name