Come utilizzare il comando nohup in Linux

Pubblicato: 2022-06-25
Laptop Linux che mostra un prompt bash
fatmawati achmad zaenuri/Shutterstock.com

Il comando Linux nohup consente ai processi importanti di continuare a funzionare anche quando la finestra del terminale che li ha avviati è chiusa. Ti mostriamo come usare questo venerabile comando sul Linux di oggi.

HUP e SIGHUP

Unix, l'antenato di Linux, è stato creato prima dell'invenzione del PC. I computer erano apparecchiature grandi e costose. Le persone interagivano con loro su linee seriali localmente all'interno dello stesso edificio o in remoto tramite connessioni modem lente. Inizialmente, hanno digitato le loro istruzioni su telescriventi che sono state gradualmente sostituite da terminali stupidi.

Erano chiamati stupidi perché la potenza di elaborazione era nel computer a cui eri collegato, non nel terminale su cui stavi digitando. I programmi erano in esecuzione sul computer, ovunque si trovasse, e non sul dispositivo sulla scrivania.

Se è successo qualcosa che ha interrotto la connessione tra il tuo terminale e il computer, il computer ha rilevato la caduta di linea e ha inviato un segnale HUP o riattacca ai programmi che stavi eseguendo. I programmi hanno cessato l'esecuzione quando hanno ricevuto il segnale.

Come eliminare un'applicazione desktop o un processo in background su Linux
CORRELATI Come eliminare un'applicazione desktop o un processo in background su Linux

Quella funzionalità sopravvive in Linux oggi. Sul tuo PC, una finestra di terminale è un'emulazione di un terminale fisico. Se hai processi in esecuzione che sono stati avviati da quella finestra del terminale e chiudi quella finestra, il segnale SIGHUP viene inviato ai programmi in modo che siano informati HUP e sappiano che dovrebbero terminare.

C'è un effetto a cascata che si verifica. Se i processi hanno avviato processi figlio, il SIGHUP viene trasmesso anche a loro in modo che sappiano che dovrebbero terminare.

Il comando nohup avvia i processi figlio ma si rifiuta di passare loro i segnali SIGHUP . Potrebbe sembrare un problema, ma in realtà è una funzione utile.

Il comando nohup

Se vuoi che un processo continui anche se la finestra del terminale da cui è stato lanciato è chiusa, hai bisogno di un modo per intercettare il SIGHUP in modo che il programma non lo riceva mai. (In realtà, la finestra del terminale non avvia i processi, vengono avviati dalla sessione della shell all'interno della finestra del terminale.) La soluzione semplice ed elegante a questo problema è posizionare un altro processo tra la sessione della shell e il programma e farlo il programma di livello intermedio non trasmette mai il segnale SIGHUP .

Questo è ciò che fa nohup . Avvia i programmi per te in modo che siano un processo figlio di nohup , non un processo figlio della shell. Poiché non sono un processo figlio della shell, non riceveranno direttamente un SIGHUP dalla shell. E se nohup non trasmette il SIGHUP ai suoi figli, il programma non riceverà affatto SIGHUP .

Ciò è utile quando, ad esempio, si dispone di un processo di lunga durata che è necessario eseguire fino al completamento. Se chiudi accidentalmente la finestra del terminale e la sua shell, interromperai anche il processo. L'utilizzo di nohup per avviare il processo isola il processo dal segnale nohup . Se stai lavorando in remoto su un computer tramite SSH e non vuoi che un processo sensibile venga terminato se la connessione remota fallisce, avvierai il processo sul computer remoto con nohup .

Usando nohup

Abbiamo creato un programma che non fa nulla di utile, ma funzionerà e funzionerà fino alla chiusura. Stampa l'ora sulla finestra del terminale ogni tre secondi. Si chiama long-proc per "processo lungo".

 ./proc 

Il programma long-proc che esegue una finestra di terminale

Se questo fosse un programma che ha fatto qualcosa di utile e volessimo che continuasse a funzionare anche se la finestra del terminale e la shell sono chiuse, lo avvieremo con nohup .

 nohup ./proc 

lanciare il programma long-proc da nohup

Il processo è disaccoppiato da stdin e stdout , quindi non può né ricevere alcun input né scrivere nella finestra del terminale. Inoltre, poiché è ancora in esecuzione, non si torna al prompt dei comandi. Tutto ciò che fa nohup è rendere il processo impermeabile alla chiusura del terminale. Non trasforma il processo in un'attività in background.

Ora devi riavviare solo per terminare il processo? No. Per interrompere un processo nohup che non hai avviato come processo in background, premi la combinazione di tasti Ctrl+C.

Arrestare il processo lungo con Ctrl+C

L'output del programma è stato catturato per noi in un file chiamato "nohup.out". Possiamo rivederlo con meno.

 meno nohup.out 

Aprire il file nohup.out in meno

Tutto ciò che normalmente verrebbe inviato alla finestra del terminale viene catturato nel file. Le successive esecuzioni di nohup verranno aggiunte al file "nohup.out" esistente.

L'output di long-proc scritto nel file nohup.out, visualizzato in less

Un modo più utile per eseguire il processo è nohup senza alcun problema in modo che resista alla chiusura della finestra del terminale e renderlo allo stesso tempo un'attività in background. Per fare ciò aggiungiamo una e commerciale " & " alla fine della riga di comando.

 nohup ./proc lungo & 

avviare il programma long-proc con nohup e renderlo un'attività in background

Dovrai premere "Invio" ancora una volta per tornare al prompt dei comandi. Ci viene detto che il numero del processo del processo è 1, il numero tra parentesi " [] ", e che l'ID del processo è 13115.

Possiamo usare uno di questi per terminare il processo. "Ctrl+C" non funzionerà ora perché il programma non ha alcuna associazione né con la finestra del terminale né con la shell.

Se dimentichi qual è il numero del lavoro, puoi utilizzare il comando jobs per elencare le attività in background che sono state avviate da quella finestra del terminale.

 lavori 

Elenca le attività in background che sono state avviate da una finestra del terminale

Per terminare il nostro task possiamo utilizzare il comando kill e il numero del job, preceduto da un segno di percentuale “ % “, in questo modo:

 uccidere %1

Se hai chiuso la finestra del terminale, dovrai trovare l'ID del processo e usarlo con il comando kill . Il comando pgrep troverà l'ID processo per i processi che corrispondono all'indizio di ricerca fornito. Cercheremo il nome del processo.

 pgrep long-proc 

Trovare l'ID processo di un processo per nome

Ora possiamo usare l'ID processo per terminare il processo.

 uccidere 13115 

Utilizzo del comando kill e dell'ID processo per terminare un processo

La prossima volta che premi "Invio" vieni informato che il processo è stato terminato.

Ora diamo un'occhiata a cosa non termina il processo. Lo riavvieremo, quindi chiuderemo la finestra del terminale.

 nohup ./proc 

Chiusura della finestra del terminale con un processo in esecuzione

Se apriamo una nuova finestra di terminale e cerchiamo il nostro processo con pgrep , vediamo che è ancora in esecuzione. La chiusura della finestra del terminale che ha avviato il processo non ha avuto alcun effetto.

 pgrep long-proc 

Utilizzo di pgrep per cercare un processo per nome

È possibile passare più comandi a nohup , ma di solito è meglio lanciarli separatamente. Rende più facile manipolarli come lavori in background. I comandi non verranno eseguiti contemporaneamente, verranno eseguiti uno dopo l'altro. L'esecuzione non è simultanea, è sequenziale. Per farli funzionare contemporaneamente è necessario avviarli separatamente.

Detto questo, per avviare più processi contemporaneamente, utilizzare nohup per avviare una shell Bash e utilizzare l'opzione -c (comandi) con la stringa di comandi. Utilizzare le virgolette singole “ ' ” per avvolgere l'elenco dei comandi e le doppie e commerciali “ && ” per separare i comandi.

 nohup bash -c 'ls /bin && ls /sbin' 

Avvio di due processi senza hup

Se usi less per esaminare il file "nohup.out", vedrai l'output del primo processo, quindi l'output del secondo processo.

 meno nohup.out 

Aprire il file nohup.out in meno

L'output di entrambi i comandi è stato catturato nel file "nohup.out". Non è intrecciato, l'output del secondo processo inizia solo una volta terminato il primo processo.

Il contenuto del file nohup.out in less

Se desideri utilizzare un file tuo invece di "nohup.out", puoi reindirizzare il comando nel file di tua scelta.

 nohup bash -c 'ls /bin && ls /sbin' > miofile.txt 

reindirizzando l'output dai processi a un file fornito dall'utente

Nota che il messaggio non dice più "aggiungere output a nohupo.out", dice "reindirizzamento stderr a stdout" e stiamo reindirizzando stdout al nostro file "myfile.txt".

Possiamo guardare all'interno del file "myfile.txt" con meno.

 meno miofile.txt 

Come prima, contiene l'output di entrambi i comandi.


È divertente come la storia di un'utilità a volte possa far sembrare che non avesse alcuna rilevanza per i tempi moderni. Il comando nohup è uno di quelli. Qualcosa che è stato creato per far fronte alle disconnessioni sulle linee seriali è ancora utile per gli utenti Linux di oggi su macchine incredibilmente potenti.

CORRELATI: 37 importanti comandi Linux che dovresti conoscere