|
马上注册,结交更多好友,享用更多功能^_^
您需要 登录 才可以下载或查看,没有账号?立即注册
x
问题描述:请问一下为什么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[128]; //这些[]里面的也可以用预定义控制
char author[40];
float price;
struct Date date;
char publisher[40];
};
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[i] = NULL;
}
}
void printLibrary(struct Book *library[]) //打印书籍 ,不是null,说已经申请地址了,并且输入书籍信息了。
{
int i;
for (i = 0; i < MAX_SIZE; i++)
{
if (library[i] != NULL)
{
printBook(library[i]);
putchar('\n');
}
}
}
void releaseLibrary(struct Book *library[]) //删除书籍,是不自己malloc的内存,才能free,即使用malloc的目的就是为了可以删除。之前教程
{
int i;
for (i = 0; i < MAX_SIZE; i++)
{
if (library[i] != NULL)
{
free(library[i]);
}
}
}
int main(void)
{
struct Book *library[MAX_SIZE]; //声明一个结构体指针数组,数组的每个元素都是一个结构体之指针。
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[index] = ptr; //ptr赋值给 结构体数组的第index个元素,结构体数组的每个元素都是一个结构体
index++;
putchar('\n');
}
else
{
printf("该图书馆已满,无法录入新数据!\n"); //接收到Y后在加了一个判断还增加书籍否的判断
break;
}
}
else
{
break;
}
}
printf("\n\n录入完毕,现在开始打印验证...\n\n");
printLibrary(library); //打印书籍
releaseLibrary(library); //删除所有书籍
return 0;
}
根据你提供的代码,我看到了问题所在。在你的代码中, 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。" 的提示。
如果问题已经解决,请设置最佳答案
|
|