这段代码是我的操作系统项目的一部分,项目要求制作应用并发进程的东西,我决定用两个玩家制作一个客户服务器扑克项目,我用儿子和孙子进程来确定手牌值。
代码中应用的方案如下:
1 http://f.cl.ly/items/2f3x3z1A3z1n3r2b0X0N/schema.jpg
代码的问题在于,在同一游戏中只有第一只手被正确评估,实际上第二只手是不正确的,而在第三局游戏中有一个错误,程序结束,这发生在每一个新游戏
以下是代码:
void check_hand(int suits[5],int ranks[5],int *point){
pid_t son[2];
int i,j;
for (i = 0; i < 2; i++){
son[i]=fork();
/***********************************************
straight flush son
************************************************/
if(son[i]==0 && i==0){
pid_t grandson[3];
int k;
for(k=0;k<3;k++){
grandson[k]=fork();
if(grandson[k]==0 && k==0){
exit(F_highcard(ranks));
}
else if(grandson[k]==0 && k==1){
exit(F_flush(suits));
}
else if(grandson[k]==0 && k==2){
exit(F_straight(ranks));
}
else if(grandson[k]<0){
puts("fork failed");
exit(-1);
}
}
int exit_status_straight,exit_status_flush,exit_status_highcard;
//waiting his sons
waitpid(grandson[0],&exit_status_highcard,0);
waitpid(grandson[1],&exit_status_flush,0);
waitpid(grandson[2],&exit_status_straight,0);
/**checkpoint A****/
//elaborate the exit statuses and exit with a value
}
/***********************************************
full house son
************************************************/
if(son[i]==0 && i==1){
pid_t grandson[2];
int k;
for(k=0;k<2;k++){
grandson[k]=fork();
if(grandson[k]==0 && k==0){
exit(F_n_pairs(ranks));
}
else if(grandson[k]==0 && k==1){
exit(F_tris_poker(ranks));
}
else if(grandson[k]<0){
exit(-1);
}
}
int exit_status_pair,exit_status_tris_or_poker;
waitpid(grandson[0],&exit_status_pair,0);
waitpid(grandson[1],&exit_status_tris_or_poker,0);
/**checkpoint B****/
//elaborate the exit statuses and exit with a value
}
}
if(son[i]<0){
puts("fork failed");
exit(-1);
}
}
/***********************************************
analysis exit status of his 2 sons
************************************************/
pid_t pid;
int status;
int values[10];
//initialization
for(j=0;j<10;j++)values[j]=0;
for(j=0;j<2;j++ ){
pid = wait(&status);
if(pid==son[0]){
values[WEXITSTATUS(status)]=1;
}
else if(pid==son[1]){
values[WEXITSTATUS(status)]=1;
}
else if(pid==-1){
puts("error");
exit(1);
}
}
for(j=9;j>=0;j--){
if(values[j]==1)break;
}
*point=j;
printf("point=%d\n",*point);
}
在下面的代码中,我添加了一些Checkpoint来查找错误,执行期间的结果是每个新游戏对于玩家1和玩家2都是相同的:
1手(始终正确)
checkpoint A
checkpoint B
point=value
2手
checkpoint A
point=value
checkpoint B
为什么会这样?父亲必须等待它的儿子,在这种情况下他不等待
3手
point=-1
and exit
提前致谢。
答案 0 :(得分:1)
我有这种强烈的感觉,你的麻烦与你上次的等待循环有关。您只等待两次子进程。无论返回的PID是否是您正在检查的孩子之一。
我的猜测是你的孙子正在等待返回,或者因为某种原因返回了值<-1;
证明将在您的等待循环中放置一个else子句,输出PID并使用创建过程的PID检查此PID。
我还要做的是是将等待循环更改为仅在两个儿子中的一个触发wait
调用返回时增加j。