Linux?C/C++?timeout命令實現(xiàn)運行具有時間限制功能
Linux C/C++ timeout命令實現(xiàn)運行具有時間限制
Linux附帶了大量命令,每個命令都是唯一的,并在特定情況下使用。Linux timeout命令的一個屬性是時間限制。可以為任何命令設(shè)置時間限制。如果時間到期,命令將停止執(zhí)行。
如何使用timeout命令
我們將解釋如何使用Linux timeout命令
timeout [OPTION] DURATION COMMAND [ARG]…
timeout [OPTION]
DURATION可以是正整數(shù)或浮點數(shù),后跟可選的單位后綴:
s - seconds (default) m - minutes h - hours d - days
未使用單位時,默認為秒。如果持續(xù)時間設(shè)置為零,則禁用關(guān)聯(lián)的超時。
其他選項
DESCRIPTION
--preserve-status
以與COMMAND相同的狀態(tài)退出,即使命令超時
--foreground
當(dāng)不直接從shell提示符運行超時時,允許COMMAND從TTY讀取并獲得TTY信號;在此模式下,COMMAND的子級不會超時
-k, --kill-after=DURATION
如果COMMAND仍在運行,也發(fā)送KILL信號在發(fā)出初始信號后很久
-s, --signal=SIGNAL
指定超時時要發(fā)送的信號;SIGNAL可以是類似“HUP”的名稱或數(shù)字;有關(guān)信號列表,請參見“kill-l”
--help
顯示此幫助并退出
--version
輸出版本信息并退出如何使用timeout命令的基本示例
1.設(shè)置定時間后終止命令:
timeout 30 ping www.baidu.com
通過使用超時,我們可以確保ping不會一直運行,占用網(wǎng)絡(luò)帶寬并糾纏任何正在ping的設(shè)備。

此命令允許ping運行五秒鐘。它正在對www.baidu.com的域名進行ping,用于研究本文的測試網(wǎng)絡(luò)上。
如果程序的執(zhí)行在超時終止之前結(jié)束,超時可以將退出代碼從程序傳遞回shell,要實現(xiàn)這一點,程序必須自動停止(換句話說,它不會因超時而終止),并且必須使用–preserve-status選項。
如果使用值為5的-c(count)選項,ping將只發(fā)出5個請求。如果我們給超時一分鐘,ping肯定會自行終止。然后我們可以使用echo檢查退出值。

2.發(fā)送正確的信號
當(dāng)timeout想要停止程序時,它會發(fā)送SIGTERM信號。這禮貌地要求程序終止。某些程序可能選擇忽略SIGTERM信號。
我們可以通過請求超時來發(fā)送SIGKILL信號來實現(xiàn)這一點??梢允褂?s(signal)選項告訴超時以發(fā)送SIGKILL信號。
timeout -s SIGKILL 20 sudo tcpdump -i ens33 -n -w 20230212.pcap

我們可以使用tcpdump 抓包的默認選項運行20秒后,發(fā)送SIGKILL信號終止進程。
3.嘗試使用SIGTERM停止程序
我們使用-k(kill after)選項。-k選項需要一個時間值作為參數(shù)。在這個命令中,我們要求超時,讓dmesg運行30秒,然后用SIGTERM信號終止它。如果dmesg在40秒后仍在運行,則意味著外交SIGTERM被忽略,超時應(yīng)發(fā)送SIGKILL以完成任務(wù)。
timeout -k 40 30 dmesg -w

dmesg運行30秒,并在收到SIGTERM信號時停止。
Linux C/C++ timeout命令實現(xiàn)
...
int main(int argc, char** argv) {
...
for(int i=1;i<argc;i++) {
char* arg = argv[i];
if(strlen(arg) <= 0) continue;
if(arg[0] == '-') {
if(!strcmp(arg, "-h") || !strcmp(arg, "--help")) {
printHelp(argv[0]);
return EXIT_SUCCESS;
} else if(!strcmp(arg, "-9") || !strcmp(arg, "--kill")) {
sig_kill = SIGKILL;
...
} else {
fprintf(stderr, "Illegal argument: %s\n", arg);
return EXIT_FAILURE;
}
} else {
/*
沒有更多選項。檢查是否有足夠的剩余參數(shù)
基本語法是/程序[OPTIONS]超時程序[ARGS]
而超時和程序是強制性的 */
if (i+2 > argc) { // 檢查是否給出超時和程序
fprintf(stderr, "Not enough arguments. Check %s --help, if you need help\n", argv[0]);
if(i+1 > argc) {
// 檢查參數(shù)是否為數(shù)字
fprintf(stderr, " Missing: TIMEOUT PROGRAM\n");
} else {
if(is_numeric(argv[i]))
fprintf(stderr, " Missing: PROGRAM\n");
else
fprintf(stderr, " Missing: TIMEOUT\n");
}
return EXIT_FAILURE;
} else {
int seconds = atoi(argv[i]);
if(seconds < 0) {
fprintf(stderr, "Timeout cannot be negative");
return EXIT_FAILURE;
}
timeout = (unsigned int)seconds;
// 合并程序和可選程序參數(shù)
for(int j=i+1;j<argc;j++)
command = strappend(command, argv[j]);
break;
}
}
}
// 檢查程序參數(shù)
if(command == NULL || strlen(command) <= 0) {
fprintf(stderr, "Not enough arguments. Check %s --help, if you need help\n", argv[0]);
fprintf(stderr, " Missing: TIMEOUT PROGRAM\n");
return EXIT_FAILURE;
}
// Fork守護程序(如果需要)
if (daemonize) {
fork_daemon();
}
...
int status;
pid_t wait_status;
// 信號處理器
signal(SIGINT, sig_handler);
signal(SIGTERM, sig_handler);
signal(SIGALRM, sig_handler);
if(verbose) printf("Child process forked with pid %d.\n", proc_pid);
// 設(shè)置報警
if(timeout > 0) alarm(timeout);
wait_status = waitpid(proc_pid, &status, 0); // Wait for child
runtime += millis();
if(wait_status < 0) {
fprintf(stderr, "Error waiting for process: %s\n", strerror(errno));
return EXIT_FAILURE;
}
status = WEXITSTATUS(status); // 獲取實際退出狀態(tài)
if(status != 0) {
if(verbose) fprintf(stderr, "Process exited with status %d after %ld milliseconds\n", status, runtime);
return status;
} else {
if(verbose) printf("Process completed after %ld milliseconds\n", runtime);
return status;
}
}
...
}
...
static void sig_handler(int sig_no) {
switch(sig_no) {
case SIGALRM:
// Timeout
if(verbose)
printf("TIMEOUT after %ld milliseconds.\n", runtime+millis());
else
printf("TIMEOUT\n");
terminate_process();
exit(EXIT_FAILURE);
break;
case SIGINT:
case SIGTERM:
if(proc_pid <= 0) exit(EXIT_FAILURE);
if(verbose) printf("Program termination request\n");
if(proc_pid > 0) kill(proc_pid, sig_no);
exit(EXIT_FAILURE);
return;
}
}
...
編譯運行

If you need the complete source code of timeout, please add WeChat number (c17865354792)?
總結(jié)
timeout是一個命令行實用程序,它運行指定的命令,如果在給定的時間段后仍在運行,則終止該命令。
擴展:Linux運行有時間限制的命令—timeout命令
當(dāng)我們想讓一個定時的crontab任務(wù)運行運行一段時間后,自動終止? 有兩種方案:
一、啟動一個進程任務(wù),然后在啟動一個殺死進程任務(wù)
二、使用linux中的timeout命令
示例:執(zhí)行crontab -e 進入定時任務(wù),添加如下命令
30 9 * * * timeout -s SIGKILL 12h /home/pirate/programs/hadoop/bin/hdfs balancer -threshold 10 > ~/balancer-stdout.log 2>~/balancer-stderr.log & 32 9 * * * /home/pirate/programs/hadoop/bin/hdfs dfsadmin -setBalancerBandwidth 304857600
到此這篇關(guān)于Linux C/C++ timeout命令實現(xiàn)(運行具有時間限制)的文章就介紹到這了,更多相關(guān)Linux運行行具有時間限制內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Qt5中QML自定義環(huán)形菜單/環(huán)形選擇框的實現(xiàn)
本文主要介紹了Qt5中QML自定義環(huán)形菜單/環(huán)形選擇框的實現(xiàn),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-03-03
C++實現(xiàn)學(xué)生管理系統(tǒng)示例解析
這篇文章主要介紹了C++實現(xiàn)學(xué)生管理系統(tǒng)示例解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
C/C++使用API實現(xiàn)數(shù)據(jù)的壓縮與解壓縮
在Windows編程中,經(jīng)常會遇到需要對數(shù)據(jù)進行壓縮和解壓縮的情況,本文將深入探討使用Windows API進行數(shù)據(jù)壓縮與解壓縮的過程,感興趣的小伙伴可以了解下2023-11-11
C++的cout.tellp()和cout.seekp()語法介紹
無論是使用 cout 輸出普通數(shù)據(jù),用 cout.put() 輸出指定字符,還是用 cout.write() 輸出指定字符串,數(shù)據(jù)都會先放到輸出流緩沖區(qū),待緩沖區(qū)刷新,數(shù)據(jù)才會輸出到指定位置,本文給大家介紹一下C++的cout.tellp()和cout.seekp()語法,需要的朋友可以參考下2023-09-09
C語言調(diào)用攝像頭實現(xiàn)生成yuv未壓縮圖片
這篇文章主要為大家詳細介紹了C語言如何調(diào)用攝像頭實現(xiàn)生成yuv未壓縮圖片,文中的示例代碼講解詳細,具有一定的學(xué)習(xí)價值,感興趣的小伙伴可以參考一下2023-11-11

