gpt4 book ai didi

c - 为什么 fgets() 工作错误?

转载 作者:行者123 更新时间:2023-11-30 15:20:32 25 4
gpt4 key购买 nike

这是我的代码

if(passwordCorretta(ds_sock)){
printf("CLIENT: password aggiungi corretta\n");

do{
printf("Cognome >> ");
fgets(cognome,sizeof(cognome),stdin);
//scanf("%s", cognome);
printf("Nome >> ");
fgets(nome,sizeof(nome),stdin);
//scanf("%s", nome);
printf("Telefono >> ");
fgets(telefono,sizeof(telefono),stdin);
//scanf("%s", telefono);

在输出中首先打印两个 printf,跳过第一个 fgets(),这是为什么?

这是我的输出

CLIENT: password aggiungi corretta
Cognome >> Nome >> **my input
Telefono >> **my input

有什么想法吗?谢谢

这是我的整个函数

void main(){
int ds_sock, length, res;
struct sockaddr_in client;
struct hostent *hp;
char oper[2];

int risPwd;
int op;

char cognome[30];
char nome[20];
char telefono[12];

char continua[3];
char ris[2];
int trovato;
int aggiunto;

ds_sock = socket(AF_INET, SOCK_STREAM, 0);

client.sin_family = AF_INET;
client.sin_port = 1999;

hp = gethostbyname("localhost"); //indirizzo del server
memcpy(&client.sin_addr, hp->h_addr, 4);

res = connect(ds_sock, &client, sizeof(client));
if(res==-1) {
perror("Errore nella connessione");
}

signal(SIGPIPE, gest_broken_pipe);
signal(SIGINT, gest_interruzione);

do{
op=scelta();
sprintf(oper,"%d",op);
if(write(ds_sock, oper, sizeof(oper))<0){
if(errno!=EINTR) perror("Errore di scrittura");
}

switch(op){
case 1:
printf("INSERIMENTO NUOVO CONTATTO\n");

if(passwordCorretta(ds_sock)){
printf("CLIENT: password aggiungi corretta\n");

do{
printf("Cognome >> ");
fgets(cognome,sizeof(cognome),stdin);
//scanf("%s", cognome);
printf("Nome >> ");
fgets(nome,sizeof(nome),stdin);
//scanf("%s", nome);
printf("Telefono >> ");
fgets(telefono,sizeof(telefono),stdin);
//scanf("%s", telefono);
if(write(ds_sock, cognome, sizeof(cognome))<0){
if(errno!=EINTR) perror("Errore di scrittura");
}
if(write(ds_sock, nome, sizeof(nome))<0){
if(errno!=EINTR) perror("Errore di scrittura");
}
if(write(ds_sock, telefono, sizeof(telefono))<0){
if(errno!=EINTR) perror("Errore di scrittura");
}

if(read(ds_sock, ris, sizeof(ris))<0){
if(errno!=EINTR) perror("Errore di lettura");
}
trovato=atoi(ris);

switch(trovato){
case 0:
printf("Errore di lettura nel Server\n");
break;
case 1:
printf("Il contatto è già presente nell'elenco\n");
break;
case 2:
if(read(ds_sock, ris, sizeof(ris))<0){
if(errno!=EINTR) perror("Errore di lettura");
}
aggiunto=atoi(ris);
if(aggiunto==0) printf("Errore di scrittura nel Server\n");
else printf("Il contatto è stato correttamente inserito nell'elenco\n");
break;
}

printf("Vuoi aggiungere un altro contatto? [SI/NO]\n");
scanf("%s", continua);
if(write(ds_sock, continua, sizeof(continua))<0){
if(errno!=EINTR) perror("Errore di scrittura");
}
}while(strcmp(continua, "SI")==0);

}
break;

case 2:
printf("RICERCA DI UN NUMERO TELEFONICO\n");

if(passwordCorretta(ds_sock)){
printf("CLIENT: password cerca corretta\n");

do{
printf("Cognome >> ");
scanf("%s", cognome);
printf("Nome >> ");
scanf("%s", nome);
if(write(ds_sock, cognome, sizeof(cognome))<0){
if(errno!=EINTR) perror("Errore di scrittura");
}
if(write(ds_sock, nome, sizeof(nome))<0){
if(errno!=EINTR) perror("Errore di scrittura");
}

if(read(ds_sock, ris, sizeof(ris))<0){
if(errno!=EINTR) perror("Errore di lettura");
}
trovato=atoi(ris);

switch(trovato){
case 0:
printf("Errore di lettura nel Server\n");
break;
case 1:
if(read(ds_sock, telefono, sizeof(telefono))<0){
if(errno!=EINTR) perror("Errore di lettura");
}
if(strcmp(telefono, "errore")==0) printf("Errore di lettura nel Server\n");
else printf("Il telefono del contatto richiesto è: %s\n", telefono);
break;
case 2:
printf("Il contatto non è presente nell'elenco\n");
break;
}

printf("Vuoi cercare un altro numero di telefono? [SI/NO]\n");
scanf("%s", continua);
if(write(ds_sock, continua, sizeof(continua))<0){
if(errno!=EINTR) perror("Errore di scrittura");
}
}while(strcmp(continua, "SI")==0);

}
break;

case 3:
printf("USCITA\n");
break;

case 0:
printf("Eseguito lo SHUTDOWN del Server\n");
break;
}

}while(op!=3 && op!=0);

close(ds_sock);

}

我使用 3 个名为 cognome、nome、telefono 的缓冲区。想法?

最佳答案

问题出在您的 scelta() 函数中。它的两个版本(使用 scanf()fgets())都是错误的。

首先我会解释一下为什么你的fgets()版本是错误的。

您将 comando 声明为 int,而 fgets() 需要一个 char *。这是未定义的行为,你很幸运(或不幸)它没有崩溃。

至于你的 scanf() 版本:这正是我在评论中预测的。 scanf() 仅读取整数输入,并将新行(ENTER 键)保留在输入缓冲区中。这意味着您在 main() 中的第一个 fgets() 调用正在读取这个剩余的 ENTER 键。

正确执行此操作的一种方法如下:

long int scelta() {
char input[20];
fgets(input, sizeof input, stdin); // this reads input
while(strchr(input, '\n') == NULL) {
// the input has not ended
// it is probably not a long int anyway but you still need to consume it
// choose your appropriate action, e.g.
while(getchar() != '\n'); // read everything in input buffer up to (and including) the ENTER key
puts("error");
fgets(input, sizeof input, stdin); // read input again
}
char *endptr;
long int comando = strtol(input, &endptr, 10);
// then check range and check for errors with endptr
return comando;
}

关于strtol()的详细描述和示例代码(带错误检查),请参阅Linux Programmer's Manual .

关于c - 为什么 fgets() 工作错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29971235/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com