日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學(xué)無先后,達(dá)者為師

網(wǎng)站首頁 編程語言 正文

Linux?C/C++?timeout命令實(shí)現(xiàn)運(yùn)行具有時(shí)間限制功能_C 語言

作者:程序猿編碼 ? 更新時(shí)間: 2023-05-08 編程語言

Linux C/C++ timeout命令實(shí)現(xiàn)運(yùn)行具有時(shí)間限制

Linux附帶了大量命令,每個(gè)命令都是唯一的,并在特定情況下使用。Linux timeout命令的一個(gè)屬性是時(shí)間限制。可以為任何命令設(shè)置時(shí)間限制。如果時(shí)間到期,命令將停止執(zhí)行。

如何使用timeout命令

我們將解釋如何使用Linux timeout命令

timeout [OPTION] DURATION COMMAND [ARG]…
timeout [OPTION]

DURATION可以是正整數(shù)或浮點(diǎn)數(shù),后跟可選的單位后綴:

s - seconds (default)
m - minutes
h - hours
d - days

未使用單位時(shí),默認(rèn)為秒。如果持續(xù)時(shí)間設(shè)置為零,則禁用關(guān)聯(lián)的超時(shí)。

其他選項(xiàng)

DESCRIPTION
       --preserve-status
              以與COMMAND相同的狀態(tài)退出,即使命令超時(shí)

       --foreground
              當(dāng)不直接從shell提示符運(yùn)行超時(shí)時(shí),允許COMMAND從TTY讀取并獲得TTY信號;在此模式下,COMMAND的子級不會超時(shí)

       -k, --kill-after=DURATION
              如果COMMAND仍在運(yùn)行,也發(fā)送KILL信號在發(fā)出初始信號后很久

       -s, --signal=SIGNAL

              指定超時(shí)時(shí)要發(fā)送的信號;SIGNAL可以是類似“HUP”的名稱或數(shù)字;有關(guān)信號列表,請參見“kill-l”

       --help
              顯示此幫助并退出 

       --version
              輸出版本信息并退出

如何使用timeout命令的基本示例

1.設(shè)置定時(shí)間后終止命令:

timeout 30 ping www.baidu.com

通過使用超時(shí),我們可以確保ping不會一直運(yùn)行,占用網(wǎng)絡(luò)帶寬并糾纏任何正在ping的設(shè)備。

此命令允許ping運(yùn)行五秒鐘。它正在對www.baidu.com的域名進(jìn)行ping,用于研究本文的測試網(wǎng)絡(luò)上。

如果程序的執(zhí)行在超時(shí)終止之前結(jié)束,超時(shí)可以將退出代碼從程序傳遞回shell,要實(shí)現(xiàn)這一點(diǎn),程序必須自動停止(換句話說,它不會因超時(shí)而終止),并且必須使用–preserve-status選項(xiàng)。

如果使用值為5的-c(count)選項(xiàng),ping將只發(fā)出5個(gè)請求。如果我們給超時(shí)一分鐘,ping肯定會自行終止。然后我們可以使用echo檢查退出值。

2.發(fā)送正確的信號

當(dāng)timeout想要停止程序時(shí),它會發(fā)送SIGTERM信號。這禮貌地要求程序終止。某些程序可能選擇忽略SIGTERM信號。

我們可以通過請求超時(shí)來發(fā)送SIGKILL信號來實(shí)現(xiàn)這一點(diǎn)。可以使用-s(signal)選項(xiàng)告訴超時(shí)以發(fā)送SIGKILL信號。

timeout -s SIGKILL 20 sudo tcpdump -i ens33 -n -w 20230212.pcap

我們可以使用tcpdump 抓包的默認(rèn)選項(xiàng)運(yùn)行20秒后,發(fā)送SIGKILL信號終止進(jìn)程。

3.嘗試使用SIGTERM停止程序

我們使用-k(kill after)選項(xiàng)。-k選項(xiàng)需要一個(gè)時(shí)間值作為參數(shù)。在這個(gè)命令中,我們要求超時(shí),讓dmesg運(yùn)行30秒,然后用SIGTERM信號終止它。如果dmesg在40秒后仍在運(yùn)行,則意味著外交SIGTERM被忽略,超時(shí)應(yīng)發(fā)送SIGKILL以完成任務(wù)。

timeout -k 40 30 dmesg -w

dmesg運(yùn)行30秒,并在收到SIGTERM信號時(shí)停止。

Linux C/C++ timeout命令實(shí)現(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 {
			/* 
			沒有更多選項(xiàng)。檢查是否有足夠的剩余參數(shù)
			基本語法是/程序[OPTIONS]超時(shí)程序[ARGS]
			而超時(shí)和程序是強(qiáng)制性的  */
    		if (i+2 > argc) {		// 檢查是否給出超時(shí)和程序
				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守護(hù)程序(如果需要)
    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è)置報(bào)警
		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);		// 獲取實(shí)際退出狀態(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;
	}
}

...

編譯運(yùn)行

If you need the complete source code of timeout, please add WeChat number (c17865354792)?

總結(jié)

timeout是一個(gè)命令行實(shí)用程序,它運(yùn)行指定的命令,如果在給定的時(shí)間段后仍在運(yùn)行,則終止該命令。

擴(kuò)展:Linux運(yùn)行有時(shí)間限制的命令—timeout命令

當(dāng)我們想讓一個(gè)定時(shí)的crontab任務(wù)運(yùn)行運(yùn)行一段時(shí)間后,自動終止? 有兩種方案:

一、啟動一個(gè)進(jìn)程任務(wù),然后在啟動一個(gè)殺死進(jìn)程任務(wù)

二、使用linux中的timeout命令

示例:執(zhí)行crontab -e 進(jìn)入定時(shí)任務(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

原文鏈接:https://blog.csdn.net/chen1415886044/article/details/128994868

欄目分類
最近更新