You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
765 lines
20 KiB
765 lines
20 KiB
3 years ago
|
import unittest
|
||
|
import weakref
|
||
|
|
||
|
from test.support import check_syntax_error, cpython_only
|
||
|
from test.support import gc_collect
|
||
|
|
||
|
|
||
|
class ScopeTests(unittest.TestCase):
|
||
|
|
||
|
def testSimpleNesting(self):
|
||
|
|
||
|
def make_adder(x):
|
||
|
def adder(y):
|
||
|
return x + y
|
||
|
return adder
|
||
|
|
||
|
inc = make_adder(1)
|
||
|
plus10 = make_adder(10)
|
||
|
|
||
|
self.assertEqual(inc(1), 2)
|
||
|
self.assertEqual(plus10(-2), 8)
|
||
|
|
||
|
def testExtraNesting(self):
|
||
|
|
||
|
def make_adder2(x):
|
||
|
def extra(): # check freevars passing through non-use scopes
|
||
|
def adder(y):
|
||
|
return x + y
|
||
|
return adder
|
||
|
return extra()
|
||
|
|
||
|
inc = make_adder2(1)
|
||
|
plus10 = make_adder2(10)
|
||
|
|
||
|
self.assertEqual(inc(1), 2)
|
||
|
self.assertEqual(plus10(-2), 8)
|
||
|
|
||
|
def testSimpleAndRebinding(self):
|
||
|
|
||
|
def make_adder3(x):
|
||
|
def adder(y):
|
||
|
return x + y
|
||
|
x = x + 1 # check tracking of assignment to x in defining scope
|
||
|
return adder
|
||
|
|
||
|
inc = make_adder3(0)
|
||
|
plus10 = make_adder3(9)
|
||
|
|
||
|
self.assertEqual(inc(1), 2)
|
||
|
self.assertEqual(plus10(-2), 8)
|
||
|
|
||
|
def testNestingGlobalNoFree(self):
|
||
|
|
||
|
def make_adder4(): # XXX add exta level of indirection
|
||
|
def nest():
|
||
|
def nest():
|
||
|
def adder(y):
|
||
|
return global_x + y # check that plain old globals work
|
||
|
return adder
|
||
|
return nest()
|
||
|
return nest()
|
||
|
|
||
|
global_x = 1
|
||
|
adder = make_adder4()
|
||
|
self.assertEqual(adder(1), 2)
|
||
|
|
||
|
global_x = 10
|
||
|
self.assertEqual(adder(-2), 8)
|
||
|
|
||
|
def testNestingThroughClass(self):
|
||
|
|
||
|
def make_adder5(x):
|
||
|
class Adder:
|
||
|
def __call__(self, y):
|
||
|
return x + y
|
||
|
return Adder()
|
||
|
|
||
|
inc = make_adder5(1)
|
||
|
plus10 = make_adder5(10)
|
||
|
|
||
|
self.assertEqual(inc(1), 2)
|
||
|
self.assertEqual(plus10(-2), 8)
|
||
|
|
||
|
def testNestingPlusFreeRefToGlobal(self):
|
||
|
|
||
|
def make_adder6(x):
|
||
|
global global_nest_x
|
||
|
def adder(y):
|
||
|
return global_nest_x + y
|
||
|
global_nest_x = x
|
||
|
return adder
|
||
|
|
||
|
inc = make_adder6(1)
|
||
|
plus10 = make_adder6(10)
|
||
|
|
||
|
self.assertEqual(inc(1), 11) # there's only one global
|
||
|
self.assertEqual(plus10(-2), 8)
|
||
|
|
||
|
def testNearestEnclosingScope(self):
|
||
|
|
||
|
def f(x):
|
||
|
def g(y):
|
||
|
x = 42 # check that this masks binding in f()
|
||
|
def h(z):
|
||
|
return x + z
|
||
|
return h
|
||
|
return g(2)
|
||
|
|
||
|
test_func = f(10)
|
||
|
self.assertEqual(test_func(5), 47)
|
||
|
|
||
|
def testMixedFreevarsAndCellvars(self):
|
||
|
|
||
|
def identity(x):
|
||
|
return x
|
||
|
|
||
|
def f(x, y, z):
|
||
|
def g(a, b, c):
|
||
|
a = a + x # 3
|
||
|
def h():
|
||
|
# z * (4 + 9)
|
||
|
# 3 * 13
|
||
|
return identity(z * (b + y))
|
||
|
y = c + z # 9
|
||
|
return h
|
||
|
return g
|
||
|
|
||
|
g = f(1, 2, 3)
|
||
|
h = g(2, 4, 6)
|
||
|
self.assertEqual(h(), 39)
|
||
|
|
||
|
def testFreeVarInMethod(self):
|
||
|
|
||
|
def test():
|
||
|
method_and_var = "var"
|
||
|
class Test:
|
||
|
def method_and_var(self):
|
||
|
return "method"
|
||
|
def test(self):
|
||
|
return method_and_var
|
||
|
def actual_global(self):
|
||
|
return str("global")
|
||
|
def str(self):
|
||
|
return str(self)
|
||
|
return Test()
|
||
|
|
||
|
t = test()
|
||
|
self.assertEqual(t.test(), "var")
|
||
|
self.assertEqual(t.method_and_var(), "method")
|
||
|
self.assertEqual(t.actual_global(), "global")
|
||
|
|
||
|
method_and_var = "var"
|
||
|
class Test:
|
||
|
# this class is not nested, so the rules are different
|
||
|
def method_and_var(self):
|
||
|
return "method"
|
||
|
def test(self):
|
||
|
return method_and_var
|
||
|
def actual_global(self):
|
||
|
return str("global")
|
||
|
def str(self):
|
||
|
return str(self)
|
||
|
|
||
|
t = Test()
|
||
|
self.assertEqual(t.test(), "var")
|
||
|
self.assertEqual(t.method_and_var(), "method")
|
||
|
self.assertEqual(t.actual_global(), "global")
|
||
|
|
||
|
def testCellIsKwonlyArg(self):
|
||
|
# Issue 1409: Initialisation of a cell value,
|
||
|
# when it comes from a keyword-only parameter
|
||
|
def foo(*, a=17):
|
||
|
def bar():
|
||
|
return a + 5
|
||
|
return bar() + 3
|
||
|
|
||
|
self.assertEqual(foo(a=42), 50)
|
||
|
self.assertEqual(foo(), 25)
|
||
|
|
||
|
def testRecursion(self):
|
||
|
|
||
|
def f(x):
|
||
|
def fact(n):
|
||
|
if n == 0:
|
||
|
return 1
|
||
|
else:
|
||
|
return n * fact(n - 1)
|
||
|
if x >= 0:
|
||
|
return fact(x)
|
||
|
else:
|
||
|
raise ValueError("x must be >= 0")
|
||
|
|
||
|
self.assertEqual(f(6), 720)
|
||
|
|
||
|
|
||
|
def testUnoptimizedNamespaces(self):
|
||
|
|
||
|
check_syntax_error(self, """if 1:
|
||
|
def unoptimized_clash1(strip):
|
||
|
def f(s):
|
||
|
from sys import *
|
||
|
return getrefcount(s) # ambiguity: free or local
|
||
|
return f
|
||
|
""")
|
||
|
|
||
|
check_syntax_error(self, """if 1:
|
||
|
def unoptimized_clash2():
|
||
|
from sys import *
|
||
|
def f(s):
|
||
|
return getrefcount(s) # ambiguity: global or local
|
||
|
return f
|
||
|
""")
|
||
|
|
||
|
check_syntax_error(self, """if 1:
|
||
|
def unoptimized_clash2():
|
||
|
from sys import *
|
||
|
def g():
|
||
|
def f(s):
|
||
|
return getrefcount(s) # ambiguity: global or local
|
||
|
return f
|
||
|
""")
|
||
|
|
||
|
check_syntax_error(self, """if 1:
|
||
|
def f():
|
||
|
def g():
|
||
|
from sys import *
|
||
|
return getrefcount # global or local?
|
||
|
""")
|
||
|
|
||
|
def testLambdas(self):
|
||
|
|
||
|
f1 = lambda x: lambda y: x + y
|
||
|
inc = f1(1)
|
||
|
plus10 = f1(10)
|
||
|
self.assertEqual(inc(1), 2)
|
||
|
self.assertEqual(plus10(5), 15)
|
||
|
|
||
|
f2 = lambda x: (lambda : lambda y: x + y)()
|
||
|
inc = f2(1)
|
||
|
plus10 = f2(10)
|
||
|
self.assertEqual(inc(1), 2)
|
||
|
self.assertEqual(plus10(5), 15)
|
||
|
|
||
|
f3 = lambda x: lambda y: global_x + y
|
||
|
global_x = 1
|
||
|
inc = f3(None)
|
||
|
self.assertEqual(inc(2), 3)
|
||
|
|
||
|
f8 = lambda x, y, z: lambda a, b, c: lambda : z * (b + y)
|
||
|
g = f8(1, 2, 3)
|
||
|
h = g(2, 4, 6)
|
||
|
self.assertEqual(h(), 18)
|
||
|
|
||
|
def testUnboundLocal(self):
|
||
|
|
||
|
def errorInOuter():
|
||
|
print(y)
|
||
|
def inner():
|
||
|
return y
|
||
|
y = 1
|
||
|
|
||
|
def errorInInner():
|
||
|
def inner():
|
||
|
return y
|
||
|
inner()
|
||
|
y = 1
|
||
|
|
||
|
self.assertRaises(UnboundLocalError, errorInOuter)
|
||
|
self.assertRaises(NameError, errorInInner)
|
||
|
|
||
|
def testUnboundLocal_AfterDel(self):
|
||
|
# #4617: It is now legal to delete a cell variable.
|
||
|
# The following functions must obviously compile,
|
||
|
# and give the correct error when accessing the deleted name.
|
||
|
def errorInOuter():
|
||
|
y = 1
|
||
|
del y
|
||
|
print(y)
|
||
|
def inner():
|
||
|
return y
|
||
|
|
||
|
def errorInInner():
|
||
|
def inner():
|
||
|
return y
|
||
|
y = 1
|
||
|
del y
|
||
|
inner()
|
||
|
|
||
|
self.assertRaises(UnboundLocalError, errorInOuter)
|
||
|
self.assertRaises(NameError, errorInInner)
|
||
|
|
||
|
def testUnboundLocal_AugAssign(self):
|
||
|
# test for bug #1501934: incorrect LOAD/STORE_GLOBAL generation
|
||
|
exec("""if 1:
|
||
|
global_x = 1
|
||
|
def f():
|
||
|
global_x += 1
|
||
|
try:
|
||
|
f()
|
||
|
except UnboundLocalError:
|
||
|
pass
|
||
|
else:
|
||
|
fail('scope of global_x not correctly determined')
|
||
|
""", {'fail': self.fail})
|
||
|
|
||
|
def testComplexDefinitions(self):
|
||
|
|
||
|
def makeReturner(*lst):
|
||
|
def returner():
|
||
|
return lst
|
||
|
return returner
|
||
|
|
||
|
self.assertEqual(makeReturner(1,2,3)(), (1,2,3))
|
||
|
|
||
|
def makeReturner2(**kwargs):
|
||
|
def returner():
|
||
|
return kwargs
|
||
|
return returner
|
||
|
|
||
|
self.assertEqual(makeReturner2(a=11)()['a'], 11)
|
||
|
|
||
|
def testScopeOfGlobalStmt(self):
|
||
|
# Examples posted by Samuele Pedroni to python-dev on 3/1/2001
|
||
|
|
||
|
exec("""if 1:
|
||
|
# I
|
||
|
x = 7
|
||
|
def f():
|
||
|
x = 1
|
||
|
def g():
|
||
|
global x
|
||
|
def i():
|
||
|
def h():
|
||
|
return x
|
||
|
return h()
|
||
|
return i()
|
||
|
return g()
|
||
|
self.assertEqual(f(), 7)
|
||
|
self.assertEqual(x, 7)
|
||
|
|
||
|
# II
|
||
|
x = 7
|
||
|
def f():
|
||
|
x = 1
|
||
|
def g():
|
||
|
x = 2
|
||
|
def i():
|
||
|
def h():
|
||
|
return x
|
||
|
return h()
|
||
|
return i()
|
||
|
return g()
|
||
|
self.assertEqual(f(), 2)
|
||
|
self.assertEqual(x, 7)
|
||
|
|
||
|
# III
|
||
|
x = 7
|
||
|
def f():
|
||
|
x = 1
|
||
|
def g():
|
||
|
global x
|
||
|
x = 2
|
||
|
def i():
|
||
|
def h():
|
||
|
return x
|
||
|
return h()
|
||
|
return i()
|
||
|
return g()
|
||
|
self.assertEqual(f(), 2)
|
||
|
self.assertEqual(x, 2)
|
||
|
|
||
|
# IV
|
||
|
x = 7
|
||
|
def f():
|
||
|
x = 3
|
||
|
def g():
|
||
|
global x
|
||
|
x = 2
|
||
|
def i():
|
||
|
def h():
|
||
|
return x
|
||
|
return h()
|
||
|
return i()
|
||
|
return g()
|
||
|
self.assertEqual(f(), 2)
|
||
|
self.assertEqual(x, 2)
|
||
|
|
||
|
# XXX what about global statements in class blocks?
|
||
|
# do they affect methods?
|
||
|
|
||
|
x = 12
|
||
|
class Global:
|
||
|
global x
|
||
|
x = 13
|
||
|
def set(self, val):
|
||
|
x = val
|
||
|
def get(self):
|
||
|
return x
|
||
|
|
||
|
g = Global()
|
||
|
self.assertEqual(g.get(), 13)
|
||
|
g.set(15)
|
||
|
self.assertEqual(g.get(), 13)
|
||
|
""")
|
||
|
|
||
|
def testLeaks(self):
|
||
|
|
||
|
class Foo:
|
||
|
count = 0
|
||
|
|
||
|
def __init__(self):
|
||
|
Foo.count += 1
|
||
|
|
||
|
def __del__(self):
|
||
|
Foo.count -= 1
|
||
|
|
||
|
def f1():
|
||
|
x = Foo()
|
||
|
def f2():
|
||
|
return x
|
||
|
f2()
|
||
|
|
||
|
for i in range(100):
|
||
|
f1()
|
||
|
|
||
|
gc_collect() # For PyPy or other GCs.
|
||
|
self.assertEqual(Foo.count, 0)
|
||
|
|
||
|
def testClassAndGlobal(self):
|
||
|
|
||
|
exec("""if 1:
|
||
|
def test(x):
|
||
|
class Foo:
|
||
|
global x
|
||
|
def __call__(self, y):
|
||
|
return x + y
|
||
|
return Foo()
|
||
|
|
||
|
x = 0
|
||
|
self.assertEqual(test(6)(2), 8)
|
||
|
x = -1
|
||
|
self.assertEqual(test(3)(2), 5)
|
||
|
|
||
|
looked_up_by_load_name = False
|
||
|
class X:
|
||
|
# Implicit globals inside classes are be looked up by LOAD_NAME, not
|
||
|
# LOAD_GLOBAL.
|
||
|
locals()['looked_up_by_load_name'] = True
|
||
|
passed = looked_up_by_load_name
|
||
|
|
||
|
self.assertTrue(X.passed)
|
||
|
""")
|
||
|
|
||
|
def testLocalsFunction(self):
|
||
|
|
||
|
def f(x):
|
||
|
def g(y):
|
||
|
def h(z):
|
||
|
return y + z
|
||
|
w = x + y
|
||
|
y += 3
|
||
|
return locals()
|
||
|
return g
|
||
|
|
||
|
d = f(2)(4)
|
||
|
self.assertIn('h', d)
|
||
|
del d['h']
|
||
|
self.assertEqual(d, {'x': 2, 'y': 7, 'w': 6})
|
||
|
|
||
|
def testLocalsClass(self):
|
||
|
# This test verifies that calling locals() does not pollute
|
||
|
# the local namespace of the class with free variables. Old
|
||
|
# versions of Python had a bug, where a free variable being
|
||
|
# passed through a class namespace would be inserted into
|
||
|
# locals() by locals() or exec or a trace function.
|
||
|
#
|
||
|
# The real bug lies in frame code that copies variables
|
||
|
# between fast locals and the locals dict, e.g. when executing
|
||
|
# a trace function.
|
||
|
|
||
|
def f(x):
|
||
|
class C:
|
||
|
x = 12
|
||
|
def m(self):
|
||
|
return x
|
||
|
locals()
|
||
|
return C
|
||
|
|
||
|
self.assertEqual(f(1).x, 12)
|
||
|
|
||
|
def f(x):
|
||
|
class C:
|
||
|
y = x
|
||
|
def m(self):
|
||
|
return x
|
||
|
z = list(locals())
|
||
|
return C
|
||
|
|
||
|
varnames = f(1).z
|
||
|
self.assertNotIn("x", varnames)
|
||
|
self.assertIn("y", varnames)
|
||
|
|
||
|
@cpython_only
|
||
|
def testLocalsClass_WithTrace(self):
|
||
|
# Issue23728: after the trace function returns, the locals()
|
||
|
# dictionary is used to update all variables, this used to
|
||
|
# include free variables. But in class statements, free
|
||
|
# variables are not inserted...
|
||
|
import sys
|
||
|
self.addCleanup(sys.settrace, sys.gettrace())
|
||
|
sys.settrace(lambda a,b,c:None)
|
||
|
x = 12
|
||
|
|
||
|
class C:
|
||
|
def f(self):
|
||
|
return x
|
||
|
|
||
|
self.assertEqual(x, 12) # Used to raise UnboundLocalError
|
||
|
|
||
|
def testBoundAndFree(self):
|
||
|
# var is bound and free in class
|
||
|
|
||
|
def f(x):
|
||
|
class C:
|
||
|
def m(self):
|
||
|
return x
|
||
|
a = x
|
||
|
return C
|
||
|
|
||
|
inst = f(3)()
|
||
|
self.assertEqual(inst.a, inst.m())
|
||
|
|
||
|
@cpython_only
|
||
|
def testInteractionWithTraceFunc(self):
|
||
|
|
||
|
import sys
|
||
|
def tracer(a,b,c):
|
||
|
return tracer
|
||
|
|
||
|
def adaptgetter(name, klass, getter):
|
||
|
kind, des = getter
|
||
|
if kind == 1: # AV happens when stepping from this line to next
|
||
|
if des == "":
|
||
|
des = "_%s__%s" % (klass.__name__, name)
|
||
|
return lambda obj: getattr(obj, des)
|
||
|
|
||
|
class TestClass:
|
||
|
pass
|
||
|
|
||
|
self.addCleanup(sys.settrace, sys.gettrace())
|
||
|
sys.settrace(tracer)
|
||
|
adaptgetter("foo", TestClass, (1, ""))
|
||
|
sys.settrace(None)
|
||
|
|
||
|
self.assertRaises(TypeError, sys.settrace)
|
||
|
|
||
|
def testEvalExecFreeVars(self):
|
||
|
|
||
|
def f(x):
|
||
|
return lambda: x + 1
|
||
|
|
||
|
g = f(3)
|
||
|
self.assertRaises(TypeError, eval, g.__code__)
|
||
|
|
||
|
try:
|
||
|
exec(g.__code__, {})
|
||
|
except TypeError:
|
||
|
pass
|
||
|
else:
|
||
|
self.fail("exec should have failed, because code contained free vars")
|
||
|
|
||
|
def testListCompLocalVars(self):
|
||
|
|
||
|
try:
|
||
|
print(bad)
|
||
|
except NameError:
|
||
|
pass
|
||
|
else:
|
||
|
print("bad should not be defined")
|
||
|
|
||
|
def x():
|
||
|
[bad for s in 'a b' for bad in s.split()]
|
||
|
|
||
|
x()
|
||
|
try:
|
||
|
print(bad)
|
||
|
except NameError:
|
||
|
pass
|
||
|
|
||
|
def testEvalFreeVars(self):
|
||
|
|
||
|
def f(x):
|
||
|
def g():
|
||
|
x
|
||
|
eval("x + 1")
|
||
|
return g
|
||
|
|
||
|
f(4)()
|
||
|
|
||
|
def testFreeingCell(self):
|
||
|
# Test what happens when a finalizer accesses
|
||
|
# the cell where the object was stored.
|
||
|
class Special:
|
||
|
def __del__(self):
|
||
|
nestedcell_get()
|
||
|
|
||
|
def testNonLocalFunction(self):
|
||
|
|
||
|
def f(x):
|
||
|
def inc():
|
||
|
nonlocal x
|
||
|
x += 1
|
||
|
return x
|
||
|
def dec():
|
||
|
nonlocal x
|
||
|
x -= 1
|
||
|
return x
|
||
|
return inc, dec
|
||
|
|
||
|
inc, dec = f(0)
|
||
|
self.assertEqual(inc(), 1)
|
||
|
self.assertEqual(inc(), 2)
|
||
|
self.assertEqual(dec(), 1)
|
||
|
self.assertEqual(dec(), 0)
|
||
|
|
||
|
def testNonLocalMethod(self):
|
||
|
def f(x):
|
||
|
class c:
|
||
|
def inc(self):
|
||
|
nonlocal x
|
||
|
x += 1
|
||
|
return x
|
||
|
def dec(self):
|
||
|
nonlocal x
|
||
|
x -= 1
|
||
|
return x
|
||
|
return c()
|
||
|
c = f(0)
|
||
|
self.assertEqual(c.inc(), 1)
|
||
|
self.assertEqual(c.inc(), 2)
|
||
|
self.assertEqual(c.dec(), 1)
|
||
|
self.assertEqual(c.dec(), 0)
|
||
|
|
||
|
def testGlobalInParallelNestedFunctions(self):
|
||
|
# A symbol table bug leaked the global statement from one
|
||
|
# function to other nested functions in the same block.
|
||
|
# This test verifies that a global statement in the first
|
||
|
# function does not affect the second function.
|
||
|
local_ns = {}
|
||
|
global_ns = {}
|
||
|
exec("""if 1:
|
||
|
def f():
|
||
|
y = 1
|
||
|
def g():
|
||
|
global y
|
||
|
return y
|
||
|
def h():
|
||
|
return y + 1
|
||
|
return g, h
|
||
|
y = 9
|
||
|
g, h = f()
|
||
|
result9 = g()
|
||
|
result2 = h()
|
||
|
""", local_ns, global_ns)
|
||
|
self.assertEqual(2, global_ns["result2"])
|
||
|
self.assertEqual(9, global_ns["result9"])
|
||
|
|
||
|
def testNonLocalClass(self):
|
||
|
|
||
|
def f(x):
|
||
|
class c:
|
||
|
nonlocal x
|
||
|
x += 1
|
||
|
def get(self):
|
||
|
return x
|
||
|
return c()
|
||
|
|
||
|
c = f(0)
|
||
|
self.assertEqual(c.get(), 1)
|
||
|
self.assertNotIn("x", c.__class__.__dict__)
|
||
|
|
||
|
|
||
|
def testNonLocalGenerator(self):
|
||
|
|
||
|
def f(x):
|
||
|
def g(y):
|
||
|
nonlocal x
|
||
|
for i in range(y):
|
||
|
x += 1
|
||
|
yield x
|
||
|
return g
|
||
|
|
||
|
g = f(0)
|
||
|
self.assertEqual(list(g(5)), [1, 2, 3, 4, 5])
|
||
|
|
||
|
def testNestedNonLocal(self):
|
||
|
|
||
|
def f(x):
|
||
|
def g():
|
||
|
nonlocal x
|
||
|
x -= 2
|
||
|
def h():
|
||
|
nonlocal x
|
||
|
x += 4
|
||
|
return x
|
||
|
return h
|
||
|
return g
|
||
|
|
||
|
g = f(1)
|
||
|
h = g()
|
||
|
self.assertEqual(h(), 3)
|
||
|
|
||
|
def testTopIsNotSignificant(self):
|
||
|
# See #9997.
|
||
|
def top(a):
|
||
|
pass
|
||
|
def b():
|
||
|
global a
|
||
|
|
||
|
def testClassNamespaceOverridesClosure(self):
|
||
|
# See #17853.
|
||
|
x = 42
|
||
|
class X:
|
||
|
locals()["x"] = 43
|
||
|
y = x
|
||
|
self.assertEqual(X.y, 43)
|
||
|
class X:
|
||
|
locals()["x"] = 43
|
||
|
del x
|
||
|
self.assertFalse(hasattr(X, "x"))
|
||
|
self.assertEqual(x, 42)
|
||
|
|
||
|
@cpython_only
|
||
|
def testCellLeak(self):
|
||
|
# Issue 17927.
|
||
|
#
|
||
|
# The issue was that if self was part of a cycle involving the
|
||
|
# frame of a method call, *and* the method contained a nested
|
||
|
# function referencing self, thereby forcing 'self' into a
|
||
|
# cell, setting self to None would not be enough to break the
|
||
|
# frame -- the frame had another reference to the instance,
|
||
|
# which could not be cleared by the code running in the frame
|
||
|
# (though it will be cleared when the frame is collected).
|
||
|
# Without the lambda, setting self to None is enough to break
|
||
|
# the cycle.
|
||
|
class Tester:
|
||
|
def dig(self):
|
||
|
if 0:
|
||
|
lambda: self
|
||
|
try:
|
||
|
1/0
|
||
|
except Exception as exc:
|
||
|
self.exc = exc
|
||
|
self = None # Break the cycle
|
||
|
tester = Tester()
|
||
|
tester.dig()
|
||
|
ref = weakref.ref(tester)
|
||
|
del tester
|
||
|
gc_collect() # For PyPy or other GCs.
|
||
|
self.assertIsNone(ref())
|
||
|
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
unittest.main()
|