Portál AbcLinuxu, 26. dubna 2024 13:32


Dotaz: fork a ukončení procesu

19.5.2007 19:17 filbar | skóre: 36 | blog: Denicek_programatora | Ostrava
fork a ukončení procesu
Přečteno: 1331×
Odpovědět | Admin
V C/C++ si forkem udělám proces, ale nevím jak je zrušit.
int ctenar(int client) {
	int pid=fork();
	if (pid==0) {
		pid=getpid();
		printf("Do knihovny vstoupil čtenář číslo %d\n",pid);
		char data[1000];
		while (1) {
			read(client,data,sizeof(data));
			if (!strncasecmp(data,"close",5)) {
					printf("Čtenář odešel\n");
					close(client);
					client=0;
					int re=kill(pid,SIGSTOP);
					if(re==-1) {
						printf("Ukončení potomka se nepovedlo\n");
					} else {
						printf("Potomek úspěšně ukončen\n");
					}

			}
		}
	} else {
		return 0;
	}
}
Chtěl bych, aby po tom co klient zadá close se proces ukončil, ale nedaří se mi to. Buď mi proces zůstane, nebo se z něho stane zoombie. Co dělám špatně?
Nástroje: Začni sledovat (1) ?Zašle upozornění na váš email při vložení nového komentáře.

Odpovědi

19.5.2007 19:24 Michal Kubeček | skóre: 72 | Luštěnice
Rozbalit Rozbalit vše Re: fork a ukončení procesu
Odpovědět | | Sbalit | Link | Blokovat | Admin
Rodič si musí přečíst návratový kód - viz funkce waitpid(). Obvykle se tak děje v reakci na signál CHLD.
19.5.2007 21:22 filbar | skóre: 36 | blog: Denicek_programatora | Ostrava
Rozbalit Rozbalit vše Re: fork a ukončení procesu
Předělal jsem to s waitpidem, a pořád mi zůstávají zoombie:-(
int ctenar(int client) {
	int pid=fork();
	if (pid==0) {
		pid=getpid();
		printf("Do knihovny vstoupil čtenář číslo %d\n",pid);
		char data[1000];
		while (1) {
			read(client,data,sizeof(data));
			if (!strncasecmp(data,"close",5)) {
					printf("Čtenář odešel\n");
					close(client);
					client=0;
					exit(0);
			}
		}
	} else {
		int status;
		int ret=waitpid(pid,&status,WNOHANG);
		if (ret==-1) {
			printf("Proces se nepodařilo ukončit\n");
		} else if (ret == 0) {
			printf("Proces běží\n");
		} else {
			printf("Proces úspěšně ukončen\n");
		}
		return 0;
	}
}
michich avatar 19.5.2007 21:30 michich | skóre: 51 | blog: ohrivane_parky
Rozbalit Rozbalit vše Re: fork a ukončení procesu
int ret=waitpid(pid,&status,WNOHANG);
Přečti si v manuálu waitpid, co znamená flag WNOHANG a bude ti jasné proč.
19.5.2007 21:38 filbar | skóre: 36 | blog: Denicek_programatora | Ostrava
Rozbalit Rozbalit vše Re: fork a ukončení procesu
WNOHANG přece znamená, že se program neblokuje. Tak jaký flag tam mám použít? Když tam dám 0, tak se celý program zastaví a nejde spustit žádné další vlákno:-(:-(:-(
michich avatar 19.5.2007 21:43 michich | skóre: 51 | blog: ohrivane_parky
Rozbalit Rozbalit vše Re: fork a ukončení procesu
Že ti to vypíše vždycky "Proces běží"? Protože to waitpid voláš moc brzo, dříve než ten klient skončí. V takovém případě waitpid nemá vůbec žádný efekt. A později, až ten potomek opravdu skončí, tak už waitpid nikdo nezavolá, tudíž z něj zůstane zombie. Tak odchytávej SIGCHLD a wait volej v handleru signálu.
19.5.2007 21:44 Michal Kubeček | skóre: 72 | Luštěnice
Rozbalit Rozbalit vše Re: fork a ukončení procesu
Pokud nechcete čekat, nevolejte waitpid() dříve, než dostanete signál CHLD.
19.5.2007 23:42 qk | skóre: 15 | blog: qk_develop
Rozbalit Rozbalit vše Re: fork a ukončení procesu
Odpovědět | | Sbalit | Link | Blokovat | Admin
no vidim i dalsi nez navrhnovane moznosti, zalezi na tom co je potreba.

Pokud nepotrebujete vedet jak syn skoncil, tak o tom je neco v norme (viz
POSIX.1-2001 specifies that if the disposition of SIGCHLD is set to SIG_IGN or the SA_NOCLDWAIT flag is set for SIGCHLD (see sigaction(2)), then children that terminate do not become zombies and a call to wait() or waitpid() will block until all children have terminated, and then fail with errno set to ECHILD.
pokud je program kratky (co do trvani, tedy ne nejaky dlouhotrvajici process) tak se da vyuzit i toho, ze po skonceni rodice, se zombici prevedou na init kterej ceka ve waitu na jejich konec a tim se odstrani zombie.

u delsiho programu, kde je potreba znat jak syn skoncil, je podle mne nejlepsi vyse navrhovane reseni.
20.5.2007 03:23 Michal Kubeček | skóre: 72 | Luštěnice
Rozbalit Rozbalit vše Re: fork a ukončení procesu

V praxi je to ale trochu složitější:

POSIX.1-1990 disallowed setting the action for SIGCHLD to SIG_IGN. POSIX.1-2001 allows this possibility, so that ignoring SIGCHLD can be used to prevent the creation of zombies (see wait(2)). Nevertheless, the historical BSD and System V behaviours for ignoring SIGCHLD differ, so that the only completely portable method of ensuring that terminated children do not become zombies is to catch the SIGCHLD signal and perform a wait(2) or similar.

Bohužel se mi teď nepodařilo dohledat, od kdy se Linux chová podle novější specifikace.

Založit nové vláknoNahoru

Tiskni Sdílej: Linkuj Jaggni to Vybrali.sme.sk Google Del.icio.us Facebook

ISSN 1214-1267, (c) 1999-2007 Stickfish s.r.o.