Using variables in signal handler - require global?

后端 未结 5 1771
不思量自难忘°
不思量自难忘° 2021-01-01 10:10

I have a signal handler to handle ctrl-c interrupt. If in the signal handler I want to read a variable set in my main script, is there an alternative to using a \"global\" s

相关标签:
5条回答
  • 2021-01-01 10:58

    If you're just reading the variable, there should be no need to make the variable "global"

    def foo():
        print a
    a = 3
    foo()  #3
    

    global is necessary to allow you to change the variable and have that change propagate into the module namespace.

    If you want to pass some state to your callback without using global, the typical way to do this us to use an instance method as the callback:

    class foo(object):
         def __init__(self,arg):
             self.arg = arg
         def callback_print_arg(self):
             print self.arg
    
    def call_callback(callback):
        callback()
    
    a = foo(42)
    call_callback(a.callback_print_arg) #42
    
    0 讨论(0)
  • 2021-01-01 11:02

    You can use a closure as the signal handler that acquires its state from the main script:

    import signal
    import sys
    import time
    
    def main_function():
    
        data_for_signal_handler = 10
    
        def signal_handler(*args):
            print data_for_signal_handler
            sys.exit()
    
        signal.signal(signal.SIGINT, signal_handler) # Or whatever signal
    
        while True:
            data_for_signal_handler += 1
            time.sleep(0.5)
    
    if __name__ == '__main__':
        main_function()
    
    0 讨论(0)
  • 2021-01-01 11:04

    Within the object-oriented paradigm (OOP) it's quite convenient to use lambdas for that purpose. Using lambdas you could pass some additional context (like a self reference) and/or get rid of the unused arguments (signal, frame).

    import time
    import signal
    
    class Application:
    
        def __init__( self ):
            signal.signal( signal.SIGINT, lambda signal, frame: self._signal_handler() )
            self.terminated = False
    
        def _signal_handler( self ):
            self.terminated = True
    
        def MainLoop( self ):        
            while not self.terminated:
                print( "I'm just doing my job like everyone else" )
                time.sleep( 3 )
    
    app = Application()
    app.MainLoop()
    
    print( "The app is terminated, exiting ..." )
    
    0 讨论(0)
  • 2021-01-01 11:04

    You can access outer-scope variables from within an inline-defined function, like so:

    my_values = {'foo':'bar'}
    def handler(signum, frame):
        for key,val in my_values.items():
            print key,val
        my_values['bat']='baz'
        #remember to use mutable types, like dicts or lists
    
    signal.signal(signal.SIGINT, handler)
    
    0 讨论(0)
  • 2021-01-01 11:14

    You can use partial to create a "closure".

    import signal
    from functools import partial
    
    def signal_handler(g_var, signal, frame):
        print "in sig handler - g_var=%s" % g_var
    
    def main():
        g_var = "test"
        signal.signal(signal.SIGINT, partial(signal_handler, g_var))
    
        time.sleep(120)
    
    
    if __name__ == '__main__':
        main()
    
    0 讨论(0)
提交回复
热议问题