常德水鱼村 发表于 2023-9-24 15:58:08

高手帮忙接解答一下请问为什么要打印两次?

问题描述:请问一下为什么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;
}

isdkz 发表于 2023-9-24 15:58:17

do while循环体内部有一个if语句,用于判断输入的字符是否是Y或者N,如果不是则会打印一次“请输入Y或N。”,然后再次执行do while循环,等待用户重新输入。因此,如果用户第一次输入的字符不是Y或者N,会打印一次提示信息,然后再次等待用户输入,这就是为什么要打印两次的原因。

Mike_python小 发表于 2023-9-24 15:58:18

问题出在这里:


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:}
回答可能有误,多多谅解!

陶远航 发表于 2023-9-24 15:58:28

根据你提供的代码,我看到了问题所在。在你的代码中,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。" 的提示。
如果问题已经解决,请设置最佳答案

常德水鱼村 发表于 2023-9-24 16:12:39

Mike_python小 发表于 2023-9-24 15:58
问题出在这里:



改成scanf("%c", &ch); 还是打印了两次提示
页: [1]
查看完整版本: 高手帮忙接解答一下请问为什么要打印两次?