高手帮忙接解答一下请问为什么要打印两次?
问题描述:请问一下为什么do while循环哪里,运行程序要打印两次“请输入Y或N。”原本目的是用户没输入Y或者N就提示用户按照要求输入Y或N
-------------------------------------------------------------------------分割线下面是源代码-----------------------------------------------------------------------------------
#include <stdio.h>
#include <stdlib.h>
#define MAX_SIZE 2
struct Date
{
int year;
int month;
int day;
};
struct Book
{
char title; //这些[]里面的也可以用预定义控制
char author;
float price;
struct Date date;
char publisher;
};
void getInput(struct Book *book); //XX指针。如函数指针,就是指向函数的指针。数组指针,就是指向数组的指针。结构体指针就是指向结构体的指针。
void printBook(struct Book *book);
void initLibrary(struct Book *library[]); //这里不是函数指针或者指针函数 ,就是简单的用指针变量做函数的参数
void printLibrary(struct Book *library[]); //用指针变量作函数参数可以将函数外部的地址传递到函数内部,使得在函数内部可以操作函数外部的数据,并且这些数据不会随着 函数的结束而被销毁。
void releaseLibrary(struct Book *library[]);
void getInput(struct Book *book) //传进来的实参是一个指针,所以要使用箭头(->)来访问结构体变量的成员。第一层要,后面层嵌套的结构体就用.就行了。
{
printf("请输入书名:");
scanf("%s", book->title);
printf("请输入作者:");
scanf("%s", book->author);
printf("请输入售价:");
scanf("%f", &book->price);
printf("请输入出版日期:");
scanf("%d-%d-%d", &book->date.year, &book->date.month, &book->date.day);
printf("请输入出版社:");
scanf("%s", book->publisher);
}
void printBook(struct Book *book)
{
printf("书名:%s\n", book->title);
printf("作者:%s\n", book->author);
printf("售价:%.2f\n", book->price);
printf("出版日期:%d-%d-%d\n", book->date.year, book->date.month, book->date.day);
printf("出版社:%s\n", book->publisher);
}
void initLibrary(struct Book *library[]) //函数的形参数是结构体指针数组,长度由预定义控制了,就不用传如长度参数了。指向什么就成为XX指针
{
int i; //这个函数好像是把结构体指针的数组都初始话为零,为什么要初始话,若不初始化,指向的不一定是自己输入的书籍信息,有可能是无效的,但是判断出的有书籍信息的就不一定全是有效值。结构体指针指向的结构体记录了书籍的信息。
for (i = 0; i < MAX_SIZE; i++)
{
library = NULL;
}
}
void printLibrary(struct Book *library[]) //打印书籍 ,不是null,说已经申请地址了,并且输入书籍信息了。
{
int i;
for (i = 0; i < MAX_SIZE; i++)
{
if (library != NULL)
{
printBook(library);
putchar('\n');
}
}
}
void releaseLibrary(struct Book *library[])//删除书籍,是不自己malloc的内存,才能free,即使用malloc的目的就是为了可以删除。之前教程
{
int i;
for (i = 0; i < MAX_SIZE; i++)
{
if (library != NULL)
{
free(library);
}
}
}
int main(void)
{
struct Book *library; //声明一个结构体指针数组,数组的每个元素都是一个结构体之指针。
struct Book *ptr = NULL; //声明一个结构体指针,指向结构体变量的指针我们称之为结构体指针。用于接收返malloc函数返回的值。
int ch, index = 0; //定义变量ch用来接收输入Y/N,定义index并初始话,作为结构体指针数组实际输入下标。
initLibrary(library);
while (1)
{
printf("请问是否需要录入图书信息(Y/N):");
do //do while循环体,会先执行“语句块”,然后再判断表达式是否为真,如果为真则继续循环;如果为假,则终止循环。执行下面的if去了
{
ch = getchar();
if(ch != 'Y' && ch != 'N') //参与运算的两个表达式只要有一个为真,结果就为真。如果等是输入的不是Y或者不是N,if就成立。,就会打印
{
printf("请输入Y或N。\n");
}
} while (ch != 'Y' && ch != 'N'); //&& 表示参与运算的两个表达式都为真时,结果才为真,否则为假。没有输入Y,没有输入N,则继续执行语句块。只有输入Y或者输入N就会跳转下面的判断了,判断是Y,N就直接break
//这里一次只是接收一个条件Y,N,非YN三种情况。没有输入Y或N就一直到do while这个内部循环里面。
if (ch == 'Y')
{
if (index < MAX_SIZE) //检查如果index小于MAX_SIZE,那么申请一个结构体内存,地址返回给prt,struct Book* 指向结构体变量的指针我们称之为结构体指针
{
ptr = (struct Book *)malloc(sizeof(struct Book)); //申请内存,返回结构体指针。为什么要自己去申请内存?原因如上。2.malloc函数返回值是无类型指针。如果函数调用失败,返回值是 NULL。
getInput(ptr); //调用getInput函数,传入的参数是ptr,结构体指针。函数作用是在ptr指向的结构体中写入书籍信息。函数里面会通过指针访问结构体。
library = ptr;//ptr赋值给 结构体数组的第index个元素,结构体数组的每个元素都是一个结构体
index++;
putchar('\n');
}
else
{
printf("该图书馆已满,无法录入新数据!\n");//接收到Y后在加了一个判断还增加书籍否的判断
break;
}
}
else
{
break;
}
}
printf("\n\n录入完毕,现在开始打印验证...\n\n");
printLibrary(library); //打印书籍
releaseLibrary(library);//删除所有书籍
return 0;
}
do while循环体内部有一个if语句,用于判断输入的字符是否是Y或者N,如果不是则会打印一次“请输入Y或N。”,然后再次执行do while循环,等待用户重新输入。因此,如果用户第一次输入的字符不是Y或者N,会打印一次提示信息,然后再次等待用户输入,这就是为什么要打印两次的原因。 问题出在这里:
ch = getchar();
if(ch != 'Y' && ch != 'N')
{
printf("请输入Y或N。\n");
}
在输入Y或N之后,会有一个换行符'\n'留在输入缓冲区中,然后进入下一次循环时,getchar()会读取到这个换行符并赋值给ch,因此条件ch != 'Y' && ch != 'N'仍然成立,会提示用户再次输入。这样就导致了多次打印提示信息的问题。
为了解决这个问题,你可以在do while循环的开头添加一行代码来清除输入缓冲区中的换行符,如下所示:
do
{
scanf("%c", &ch); // 读取单个字符,包括换行符
if (ch != 'Y' && ch != 'N')
{
printf("请输入Y或N。\n");
}
} while (ch != 'Y' && ch != 'N');
这样可以确保每次循环只读取一个有效的字符,避免重复打印提示信息。
球一个最佳答案谢谢啦!这对我非常重要!{:10_254:}{:10_254:}{:10_254:}
回答可能有误,多多谅解!
根据你提供的代码,我看到了问题所在。在你的代码中,getchar()函数会读取输入缓冲区中的空白字符(包括回车、换行符等),而不是等待用户输入。这就导致了在执行 do-while 循环时,它会首先读取之前输入的换行符,然后再读取用户的实际输入。
解决这个问题的方法是在 getchar()之前加一个额外的 getchar() 来消耗掉之前的换行符。修改如下:
printf("请问是否需要录入图书信息(Y/N):");
do {
getchar(); // 消耗之前的换行符
ch = getchar();
if (ch != 'Y' && ch != 'N') {
printf("请输入Y或N。\n");
}
} while (ch != 'Y' && ch != 'N');
这样修改后,程序会等待用户输入 Y 或 N,并且只会打印一次 "请输入Y或N。" 的提示。
如果问题已经解决,请设置最佳答案 Mike_python小 发表于 2023-9-24 15:58
问题出在这里:
改成scanf("%c", &ch); 还是打印了两次提示
页:
[1]