summaryrefslogtreecommitdiff
path: root/Lib/threading.py
diff options
context:
space:
mode:
authorTim Peters <tim.peters@gmail.com>2004-07-21 03:36:52 +0000
committerTim Peters <tim.peters@gmail.com>2004-07-21 03:36:52 +0000
commit21429932e4f1c31a25c943853eb6c968b5cf9a82 (patch)
tree81b6d54ab2654895894748362bd229483523dc0a /Lib/threading.py
parent090e636add33907d196b9899eb0f019654a055e8 (diff)
downloadcpython-git-21429932e4f1c31a25c943853eb6c968b5cf9a82.tar.gz
Thread.__delete: Discussion of internal obscurities belongs in comments
rather than in docstrings. Rewrote so that _active_limbo_lock is released no matter what happens (it could have been left locked if _sys got None'd out). Use "in" in preference to has_key() for dict lookup. Don't bother looking for 'dummy_threading' in sys.modules unless KeyError is raised. Since the heart of the method is the del, do that in only one place.
Diffstat (limited to 'Lib/threading.py')
-rw-r--r--Lib/threading.py59
1 files changed, 28 insertions, 31 deletions
diff --git a/Lib/threading.py b/Lib/threading.py
index 2e6fab9f9b..860c2fdd33 100644
--- a/Lib/threading.py
+++ b/Lib/threading.py
@@ -493,41 +493,38 @@ class Thread(_Verbose):
self.__block.release()
def __delete(self):
- """Remove the current thread from the dict of currently running
- threads.
-
- Must take care to not raise an exception if dummy_thread is being used
- (and thus this module is being used as an instance of dummy_threading).
- Since dummy_thread.get_ident() always returns -1 since there is only one
- thread if dummy_thread is being used. This means that if any Thread
- instances are created they will overwrite any other threads registered.
-
- This is an issue with this method, though, since an instance of
- _MainThread is always created by 'threading'. This gets overwritten the
- instant an instance of Thread is created; both threads will have -1 as
- their value from dummy_thread.get_ident() and thus have the same key in
- the dict. This means that when the _MainThread instance created by
- 'threading' tries to clean itself up when atexit calls this method it
- gets a key error if another Thread instance was created since that
- removed the only thing with the key of -1.
-
- This all means that KeyError from trying to delete something from
- _active if dummy_threading is being used is a red herring. But since
- it isn't if dummy_threading is *not* being used then don't hide the
- exception. Also don't need to worry about issues from interpreter
- shutdown and sys not being defined because the call is protected by a
- blanket try/except block where that could be a problem.
-
- """
+ "Remove current thread from the dict of currently running threads."
+
+ # Notes about running with dummy_thread:
+ #
+ # Must take care to not raise an exception if dummy_thread is being
+ # used (and thus this module is being used as an instance of
+ # dummy_threading). dummy_thread.get_ident() always returns -1 since
+ # there is only one thread if dummy_thread is being used. Thus
+ # len(_active) is always <= 1 here, and any Thread instance created
+ # overwrites the (if any) thread currently registered in _active.
+ #
+ # An instance of _MainThread is always created by 'threading'. This
+ # gets overwritten the instant an instance of Thread is created; both
+ # threads return -1 from dummy_thread.get_ident() and thus have the
+ # same key in the dict. So when the _MainThread instance created by
+ # 'threading' tries to clean itself up when atexit calls this method
+ # it gets a KeyError if another Thread instance was created.
+ #
+ # This all means that KeyError from trying to delete something from
+ # _active if dummy_threading is being used is a red herring. But
+ # since it isn't if dummy_threading is *not* being used then don't
+ # hide the exception.
+
_active_limbo_lock.acquire()
- if _sys.modules.has_key('dummy_threading'):
+ try:
try:
del _active[_get_ident()]
except KeyError:
- pass
- else:
- del _active[_get_ident()]
- _active_limbo_lock.release()
+ if 'dummy_threading' not in _sys.modules:
+ raise
+ finally:
+ _active_limbo_lock.release()
def join(self, timeout=None):
assert self.__initialized, "Thread.__init__() not called"