Heim >Backend-Entwicklung >Python-Tutorial >Python verwendet das Signalmodul, um die geplante Ausführung zu implementieren

Python verwendet das Signalmodul, um die geplante Ausführung zu implementieren

高洛峰
高洛峰Original
2016-10-17 16:51:401481Durchsuche

Wenn Sie im Liunx-System jede Minute einen Befehl ausführen möchten, ist Crontab die häufigste Methode. Wenn Sie Crontab nicht verwenden möchten, haben meine Kollegen darauf hingewiesen, dass Sie diese Funktion mit einem Timer implementieren können Im Programm, also begann ich zu erkunden, stellte ich fest, dass ich etwas Wissen über Signale benötige...

Überprüfen Sie, welche Signale Ihr Linux unterstützt: kill -l

root@server:~# kill -l
 1) SIGHUP       2) SIGINT       3) SIGQUIT      4) SIGILL       5) SIGTRAP
 6) SIGABRT      7) SIGBUS       8) SIGFPE       9) SIGKILL     10) SIGUSR1
11) SIGSEGV     12) SIGUSR2     13) SIGPIPE     14) SIGALRM     15) SIGTERM
16) SIGSTKFLT   17) SIGCHLD     18) SIGCONT     19) SIGSTOP     20) SIGTSTP
21) SIGTTIN     22) SIGTTOU     23) SIGURG      24) SIGXCPU     25) SIGXFSZ
26) SIGVTALRM   27) SIGPROF     28) SIGWINCH    29) SIGIO       30) SIGPWR
31) SIGSYS      34) SIGRTMIN    35) SIGRTMIN+1  36) SIGRTMIN+2  37) SIGRTMIN+3
38) SIGRTMIN+4  39) SIGRTMIN+5  40) SIGRTMIN+6  41) SIGRTMIN+7  42) SIGRTMIN+8
43) SIGRTMIN+9  44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12 47) SIGRTMIN+13
48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14 51) SIGRTMAX-13 52) SIGRTMAX-12
53) SIGRTMAX-11 54) SIGRTMAX-10 55) SIGRTMAX-9  56) SIGRTMAX-8  57) SIGRTMAX-7
58) SIGRTMAX-6  59) SIGRTMAX-5  60) SIGRTMAX-4  61) SIGRTMAX-3  62) SIGRTMAX-2
63) SIGRTMAX-1  64) SIGRTMAX
root@server:~#

Signal: eine Methode zur Kommunikation zwischen Prozessen. Sobald ein Prozess ein Signal empfängt, unterbricht er den ursprünglichen Programmausführungsablauf, um das Signal zu verarbeiten. Das Betriebssystem legt das Standardverhalten des Prozesses nach dem Empfang des Signals fest. Wir können jedoch das Verhalten des Prozesses nach dem Empfang des Signals ändern, indem wir die Signalverarbeitungsfunktion binden. Es gibt zwei Signale, die nicht geändert werden können: SIGTOP und SIGKILL.

Es gibt im Allgemeinen zwei Gründe für das Senden von Signalen:

1 (passiv) Der Kernel erkennt ein Systemereignis. Wenn beispielsweise der untergeordnete Prozess beendet wird, sendet er das SIGCHLD-Signal an den übergeordneten Prozess Durch Drücken von Strg c auf der Tastatur wird ein SIGINT-Signal gesendet.

2 (aktiv) sendet ein Signal an den angegebenen Prozess über den Systemaufruf kill

Es gibt eine Setitimer-Funktion in der Sprache C Die Funktion setitimer kann drei voneinander unabhängige Timer bereitstellen. Bei jedem zeitgesteuerten Abschluss wird ein Zeitsignal an den Prozess gesendet und die Zeit automatisch neu eingestellt. Der Parameter, der den Typ des Timers bestimmt:

ITIMER_REAL Timing in Echtzeit, identisch mit dem Alarmtyp. SIGALRM

ITIMER_VIRT Die tatsächliche Ausführungszeit des geplanten Prozesses im Benutzermodus. SIGVTALRM

ITIMER_PROF Die tatsächliche Ausführungszeit des geplanten Prozesses im Benutzermodus und Kernmodus. SIGPROF

Diese drei Timer senden unterschiedliche Signale an den Prozess, wenn das Timing abgeschlossen ist. Darunter sendet der ITIMER_REAL-Klassen-Timer das SIGALRM-Signal, der ITIMER_VIRT-Klassen-Timer das SIGVTALRM-Signal und der ITIMER_REAL-Klassen-Timer SIGPROF-Signal.

Die Alarmfunktion stellt im Wesentlichen einen nicht überlasteten ITIMER_REAL-Timer mit geringer Genauigkeit ein. Er kann nur auf Sekunden genau sein und jede Einstellung kann nur einen Timer generieren. Die von der Funktion setitimer eingestellten Timer sind unterschiedlich. Sie können die Zeit nicht nur auf Mikrosekunden einstellen (theoretisch), sondern auch die Zeit automatisch ändern. In einem Unix-Prozess können Alarm- und ITIMER_REAL-Timer nicht gleichzeitig verwendet werden.

SIGINT Prozess beenden Prozess unterbrechen (Steuerung c)

SIGTERM Prozess beenden Software-Beendigungssignal

SIGKILL Prozess beenden Prozess beenden

SIGALRM Weckersignal

Die Vorkenntnisse sind fast fertig und es ist Zeit, sich mit dem Python-Signal zu befassen.

Signalnamen definieren

Das Signalpaket definiert jeden Signalnamen und seine entsprechende Ganzzahl. Zum Beispiel

import signal
print signal.SIGALRM
print signal.SIGCONT

Die von Python verwendeten Signalnamen sind dieselben Wird von Linux verwendet. Sie können die

$man 7 signal
voreingestellte Signalverarbeitungsfunktion über

abfragen. Der Kern des Signalpakets besteht darin, die Funktion signal.signal() zum Voreinstellen (Registrieren) zu verwenden ) Signalverarbeitungsfunktion, wie unten gezeigt:

singnal.signal(signalnum, handler)

signalnum ist ein Signal und handler ist die Verarbeitungsfunktion des Signals. Wir haben in Signal Basics erwähnt, dass ein Prozess Signale ignorieren, Standardaktionen ausführen oder Aktionen anpassen kann. Wenn der Handler signal.SIG_IGN ist, wird das Signal ignoriert. Wenn der Handler singal.SIG_DFL ist, führt der Prozess die Standardaktion (Standard) aus. Wenn der Handler ein Funktionsname ist, führt der Prozess die in der Funktion definierte Aktion aus.

import signal
# Define signal handler function
def myHandler(signum, frame):
  print('I received: ', signum)
  
# register signal.SIGTSTP's handler 
signal.signal(signal.SIGTSTP, myHandler)
signal.pause()
print('End of Signal Demo')

Im Hauptprogramm verwenden wir zunächst die Funktion signal.signal(), um die Signalverarbeitungsfunktion voreinzustellen. Dann führen wir signal.pause() aus, um den Prozess anzuhalten und auf das Signal zu warten. Wenn das Signal SIGUSR1 an den Prozess übergeben wird, wird der Prozess aus der Pause fortgesetzt und führt die SIGTSTP-Signalverarbeitungsfunktion myHandler() gemäß der Standardeinstellung aus. Einer der beiden Parameter von myHandler wird verwendet, um das Signal zu identifizieren (Signum), und der andere wird verwendet, um den Status des Prozessstapels (Stack-Frame) zu erhalten, wenn das Signal auftritt. Beide Parameter werden von der Funktion signal.singnal() übergeben.

Das obige Programm kann in einer Datei (z. B. test.py) gespeichert werden. Wir verwenden zum Ausführen die folgende Methode:

$python test.py

, um den Prozess laufen zu lassen. Wenn das Programm signal.pause() erreicht, pausiert der Prozess und wartet auf das Signal. Senden Sie an dieser Stelle das SIGTSTP-Signal an den Prozess, indem Sie STRG Z drücken. Wir können sehen, dass der Prozess die Funktion myHandle() ausführt, dann zum Hauptprogramm zurückkehrt und die Ausführung fortsetzt. (Natürlich können Sie auch $ps verwenden, um die Prozess-ID abzufragen, und dann $kill verwenden, um ein Signal zu senden.)

(Der Prozess muss nicht signal.pause() verwenden, um anzuhalten und zu warten für das Signal kann es auch funktionieren. Signale empfangen in, zum Beispiel die obige signal.pause() in eine Schleife umwandeln, deren Arbeit lange dauert)

Wir können die Operationen in myHandler() ändern entsprechend unseren Bedürfnissen, um unterschiedliche Signale zu implementieren.

SIGALRM-Signal regelmäßig senden

Eine nützliche Funktion ist signal.alarm(), die verwendet wird, um nach einer bestimmten Zeit ein SIGALRM-Signal an den Prozess selbst zu senden:

import signal
# Define signal handler function
def myHandler(signum, frame):
  print("Now, it's the time")
  exit()
  
# register signal.SIGALRM's handler 
signal.signal(signal.SIGALRM, myHandler)
signal.alarm(5)
while True:
  print('not yet')

Wir verwenden hier eine Endlosschleife, um den Prozess am Laufen zu halten. Nachdem signal.alarm() 5 Sekunden lang ausgeführt wurde, sendet der Prozess das SIGALRM-Signal an sich selbst und beginnt dann mit der Ausführung der Signalverarbeitungsfunktion myHandler.

发送信号

signal包的核心是设置信号处理函数。除了signal.alarm()向自身发送信号之外,并没有其他发送信号的功能。但在os包中,有类似于linux的kill命令的函数,分别为

os.kill(pid, sid)
os.killpg(pgid, sid)

分别向进程和进程组(见Linux进程关系)发送信号。sid为信号所对应的整数或者singal.SIG*。

实际上signal, pause,kill和alarm都是Linux应用编程中常见的C库函数,在这里,我们只不过是用Python语言来实现了一下。实际上,Python 的解释器是使用C语言来编写的,所以有此相似性也并不意外。此外,在Python 3.4中,signal包被增强,信号阻塞等功能被加入到该包中。我们暂时不深入到该包中。


Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn