Added the atexit module and documentation from Skip Montanaro
<skip@mojam.com>. Revisions to the markup to make it pass LaTeX, added an index entry and a reference from the sys.exitfunc documentation. This closes SourceForge patch #100620.
This commit is contained in:
parent
a05e293a21
commit
c19425d520
@ -228,6 +228,7 @@ LIBFILES= $(MANSTYLES) $(COMMONTEX) \
|
||||
../lib/libstatvfs.tex \
|
||||
../lib/libtty.tex \
|
||||
../lib/libasyncore.tex \
|
||||
../lib/libatexit.tex \
|
||||
../lib/libcfgparser.tex
|
||||
|
||||
# LaTeX source files for Macintosh Library Modules.
|
||||
|
@ -73,6 +73,7 @@ and how to embed it in other applications.
|
||||
|
||||
\input{libpython} % Python Services
|
||||
\input{libsys}
|
||||
\input{libatexit}
|
||||
\input{libtypes}
|
||||
\input{libuserdict}
|
||||
\input{liboperator}
|
||||
|
59
Doc/lib/libatexit.tex
Normal file
59
Doc/lib/libatexit.tex
Normal file
@ -0,0 +1,59 @@
|
||||
\section{\module{atexit} ---
|
||||
exit handlers}
|
||||
|
||||
\declaremodule{standard}{atexit}
|
||||
\moduleauthor{Skip Montanaro}{skip@mojam.com}
|
||||
\sectionauthor{Skip Montanaro}{skip@mojam.com}
|
||||
\modulesynopsis{Register and execute cleanup functions.}
|
||||
|
||||
The \module{atexit} module defines a single function to register
|
||||
cleanup functions. Functions thus registered are automatically
|
||||
executed upon normal interpreter termination.
|
||||
|
||||
Note: the functions registered via this module are not called when the program is killed by a
|
||||
signal, when a Python fatal internal error is detected, or when
|
||||
\code{os._exit()} is called.
|
||||
|
||||
This is an alternate interface to the functionality provided by the
|
||||
\code{sys.exitfunc} variable.
|
||||
\withsubitem{(in sys)}{\ttindex{exitfunc}}
|
||||
|
||||
\begin{funcdesc}{register}{func\optional{, *args\optional{, **kargs}}}
|
||||
Register \var{func} as a function to be executed at termination. Any
|
||||
optional arguments that are to be passed to \var{func} must be passed
|
||||
as arguments to \function{register()}.
|
||||
|
||||
At normal program termination (for instance, if
|
||||
\function{sys.exit()} is called or the main module's execution
|
||||
completes), all functions registered are called in last in, first out
|
||||
order. The assumption is that lower level modules will normally be
|
||||
imported before higher level modules and thus must be cleaned up
|
||||
later.
|
||||
\end{funcdesc}
|
||||
|
||||
|
||||
\subsection{\module{atexit} Example \label{atexit-example}}
|
||||
|
||||
The following simple example demonstrates how a module can initialize
|
||||
a counter from a file when it is imported and save the counter's
|
||||
updated value automatically when the program terminates without
|
||||
relying on the application making an explicit call into this module at
|
||||
termination.
|
||||
|
||||
\begin{verbatim}
|
||||
try:
|
||||
_count = int(open("/tmp/counter").read())
|
||||
except IOError:
|
||||
_count = 0
|
||||
|
||||
def incrcounter(n):
|
||||
global _count
|
||||
_count = _count + n
|
||||
|
||||
def savecounter():
|
||||
open("/tmp/counter", "w").write("%d" % _count)
|
||||
|
||||
import atexit
|
||||
atexit.register(savecounter)
|
||||
\end{verbatim}
|
||||
|
@ -127,7 +127,9 @@ way to exit a program when an error occurs.
|
||||
This value is not actually defined by the module, but can be set by
|
||||
the user (or by a program) to specify a clean-up action at program
|
||||
exit. When set, it should be a parameterless function. This function
|
||||
will be called when the interpreter exits. Note: the exit function
|
||||
will be called when the interpreter exits. Only one function may be
|
||||
installed in this way; to allow multiple functions which will be called
|
||||
at termination, use the \refmodule{atexit} module. Note: the exit function
|
||||
is not called when the program is killed by a signal, when a Python
|
||||
fatal internal error is detected, or when \code{os._exit()} is called.
|
||||
\end{datadesc}
|
||||
|
54
Lib/atexit.py
Normal file
54
Lib/atexit.py
Normal file
@ -0,0 +1,54 @@
|
||||
"""
|
||||
atexit.py - allow programmer to define multiple exit functions to be executed
|
||||
upon normal program termination.
|
||||
|
||||
One public function, register, is defined.
|
||||
"""
|
||||
|
||||
_exithandlers = []
|
||||
def _run_exitfuncs():
|
||||
"""run any registered exit functions
|
||||
|
||||
_exithandlers is traversed in reverse order so functions are executed
|
||||
last in, first out.
|
||||
"""
|
||||
|
||||
while _exithandlers:
|
||||
func, targs, kargs = _exithandlers[-1]
|
||||
apply(func, targs, kargs)
|
||||
_exithandlers.remove(_exithandlers[-1])
|
||||
|
||||
def register(func, *targs, **kargs):
|
||||
"""register a function to be executed upon normal program termination
|
||||
|
||||
func - function to be called at exit
|
||||
targs - optional arguments to pass to func
|
||||
kargs - optional keyword arguments to pass to func
|
||||
"""
|
||||
_exithandlers.append((func, targs, kargs))
|
||||
|
||||
import sys
|
||||
try:
|
||||
x = sys.exitfunc
|
||||
except AttributeError:
|
||||
sys.exitfunc = _run_exitfuncs
|
||||
else:
|
||||
# if x isn't our own exit func executive, assume it's another
|
||||
# registered exit function - append it to our list...
|
||||
if x != _run_exitfuncs:
|
||||
register(x)
|
||||
del sys
|
||||
|
||||
if __name__ == "__main__":
|
||||
def x1():
|
||||
print "running x1"
|
||||
def x2(n):
|
||||
print "running x2(%s)" % `n`
|
||||
def x3(n, kwd=None):
|
||||
print "running x3(%s, kwd=%s)" % (`n`, `kwd`)
|
||||
|
||||
register(x1)
|
||||
register(x2, 12)
|
||||
register(x3, 5, "bar")
|
||||
register(x3, "no kwd args")
|
||||
|
4
Lib/test/output/test_atexit
Normal file
4
Lib/test/output/test_atexit
Normal file
@ -0,0 +1,4 @@
|
||||
test_atexit
|
||||
handler2 (7,) {'kw': 'abc'}
|
||||
handler2 () {}
|
||||
handler1
|
24
Lib/test/test_atexit.py
Normal file
24
Lib/test/test_atexit.py
Normal file
@ -0,0 +1,24 @@
|
||||
# Test the exit module
|
||||
from test_support import verbose
|
||||
import atexit
|
||||
|
||||
def handler1():
|
||||
print "handler1"
|
||||
|
||||
def handler2(*args, **kargs):
|
||||
print "handler2", args, kargs
|
||||
|
||||
# save any exit functions that may have been registered as part of the
|
||||
# test framework
|
||||
_exithandlers = atexit._exithandlers
|
||||
atexit._exithandlers = []
|
||||
|
||||
atexit.register(handler1)
|
||||
atexit.register(handler2)
|
||||
atexit.register(handler2, 7, kw="abc")
|
||||
|
||||
# simulate exit behavior by calling atexit._run_exitfuncs directly...
|
||||
atexit._run_exitfuncs()
|
||||
|
||||
# restore exit handlers
|
||||
atexit._exithandlers = _exithandlers
|
Loading…
x
Reference in New Issue
Block a user