我不明白为什么我的代码输出如下。它不符合我在程序中编写的逻辑。我正在使用 System V信号量彼此同步线程。
该程序只需要标准输入到“主线程”的字符串,这些字符串应由子线程写在“路径名”传递给argv的文件上。为简单起见,我删除了这部分,只保留了scanf和printf,这完全被程序的逻辑弄得一团糟!
我该如何解决?
#define _GNU_SOURCE
#include <stdio.h>
#include <pthread.h>
#include <unistd.h>
#include <sys/syscall.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <signal.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#define fflush(stdin) while(getchar()!='\n') //REMOVE THIS
void *gestore(void *);
char buf[10];
struct info
{
pthread_t tid[100];
int v[100];
};
struct info TID;
union semun
{
int val;
struct semid_ds *buf;
unsigned short *array;
struct seminfo *__buf;
};
int indice;
int id_sem;
void *gestore(void *str)
{
int fd, i, j;
struct sembuf op[1];
printf("Thread(%ld) CREATED\n",syscall(SYS_gettid));
fflush(stdout);
fd = open((char *)(str),O_CREAT|O_RDWR,0666);
if(fd==-1)
{
printf("Error\n");
return;
}
else
{
printf("File Opened\n");
fflush(stdout);
}
repeat:
op[0].sem_num=0;
op[0].sem_op=-1;
op[0].sem_op=0;
semop(id_sem,op,1);
printf("HELLO\n");
fflush(stdout);
for(i=0; i<5; i++)
{
if(write(fd,&(buf[2*i]),1)==-1)
{
printf("ErroreWrite\n");
fflush(stdout);
return;
}
else
{
printf("WRITED\n");
fflush(stdout);
}
}
op[0].sem_num=1;
op[0].sem_op=1;
op[0].sem_op=0;
semop(id_sem,op,1);
goto repeat;
if(close(fd)==-1)
{
printf("Errore close\n");
return;
}
else
{
printf("File CLOSED\n");
fflush(stdout);
}
}
int main(int argc, char *argv[])
{
if(argc<2)
{
printf("There must be at least 2 parameters\n");
fflush(stdout);
return -1;
}
int i;
indice = argc;
printf("Number of arguments:%d \n",argc);
fflush(stdout);
//SEM SYSTEM V
union semun arg;
struct sembuf op[1];
id_sem = semget(6534,2,IPC_CREAT|0666);
arg.val=0;
semctl(id_sem,0,SETVAL,arg);
arg.val=1;
semctl(id_sem,1,SETVAL,arg);
//////////////////////
for(i=0; i<(argc-1); i++)
{
pthread_create(&(TID.tid[i]),NULL,gestore,argv[i+1]);
}
while(1)
{
op[0].sem_num=1;
op[0].sem_op=-1;
op[0].sem_op=0;
semop(id_sem,op,1);
for(i=0;i<5;i++)
{
printf("Insert value(%d): \n",i);
fflush(stdout);
scanf("%c",&(buf[2*i]));
buf[2*i+1]='\0';
fflush(stdin);//USE while(getchar()!='\n');
}
op[0].sem_num=0;
op[0].sem_op=1;
op[0].sem_op=0;
semop(id_sem,op,1);
}
}
这是我的输出: (a,b,c,d,e,f,g是我从标准输入中输入的信息)
我的字符串“ Insert Value(%d)...”在哪里?
Number of arguments:4
a
Thread(1048) CREATED
Thread(1049) CREATED
Thread(1050) CREATED
b
File OPENED
c
File OPENED
d
File OPENED
e
f
HELLO
g
HELLO
h
WRITED
i
WRITED
l
WRITED
m
WRITED
n
WRITED
o
WRITED
p
WRITED
q
WRITED
e
WRITED
r
WRITED
s
a
s
在下面的代码中,用fflus(stdin)代替了我定义的宏:while(getchar()!='N');得到的输出是:
Number of arguments:4
Thread(1090) CREATED
Thread(1091) CREATED
Thread(1092) CREATED
File OPENED
File OPENED
HELLO
File OPENED
WRITED
HELLO
WRITED
WRITED
WRITED
WRITED
WRITED
WRITED
WRITED
WRITED
WRITED
在这种情况下,我得到了这些输出而没有发送任何输入字符。
答案 0 :(得分:1)
发现了奥秘。三元组:
它以下列方式严重地写在了各处:
还与其他用户向我指出的错误定义有关,这是一场灾难。
谢谢大家!
答案 1 :(得分:0)
不知道它是否可以解决您的问题,但是您的define
确实存在代码缺陷。知道冲洗标准输入是一个很大的麻烦的读者不仅感到困惑。无论参数如何,它也将更改对fflush
的所有调用。这是一个Hello World程序来演示:
#include <stdio.h>
#define fflush(stdin) printf("Hello, World!\n")
int main(void)
{
// Will call above macro despite the fact that I'm trying to flush stdout
fflush(stdout);
}
如果要使用宏,请执行以下操作:
#define FLUSH_STDIN while(getchar()!='\n')
但是除非您有充分的理由使用宏,否则请使用函数:
void flush_stdin(void) { while(getchar()!='\n'); }
我还建议不要使用下划线(struct seminfo *__buf;
)开头的标识符,因为这些标识符保留在C中。