鱼C论坛

 找回密码
 立即注册
查看: 485|回复: 4

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

[复制链接]
发表于 2023-9-24 15:58:08 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

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;
}
最佳答案
2023-9-24 15:58:28
根据你提供的代码,我看到了问题所在。在你的代码中,getchar()函数会读取输入缓冲区中的空白字符(包括回车、换行符等),而不是等待用户输入。这就导致了在执行 do-while 循环时,它会首先读取之前输入的换行符,然后再读取用户的实际输入。

解决这个问题的方法是在 getchar()之前加一个额外的 getchar() 来消耗掉之前的换行符。修改如下:

  1. printf("请问是否需要录入图书信息(Y/N):");
  2. do {
  3.     getchar(); // 消耗之前的换行符
  4.     ch = getchar();
  5.     if (ch != 'Y' && ch != 'N') {
  6.         printf("请输入Y或N。\n");
  7.     }
  8. } while (ch != 'Y' && ch != 'N');
复制代码


这样修改后,程序会等待用户输入 Y 或 N,并且只会打印一次 "请输入Y或N。" 的提示。
如果问题已经解决,请设置最佳答案
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2023-9-24 15:58:17 | 显示全部楼层
do while循环体内部有一个if语句,用于判断输入的字符是否是Y或者N,如果不是则会打印一次“请输入Y或N。”,然后再次执行do while循环,等待用户重新输入。因此,如果用户第一次输入的字符不是Y或者N,会打印一次提示信息,然后再次等待用户输入,这就是为什么要打印两次的原因。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-9-24 15:58:18 | 显示全部楼层
问题出在这里:
  1. ch = getchar();
  2. if(ch != 'Y' && ch != 'N')
  3. {
  4.     printf("请输入Y或N。\n");
  5. }
复制代码

在输入Y或N之后,会有一个换行符'\n'留在输入缓冲区中,然后进入下一次循环时,getchar()会读取到这个换行符并赋值给ch,因此条件ch != 'Y' && ch != 'N'仍然成立,会提示用户再次输入。这样就导致了多次打印提示信息的问题。

为了解决这个问题,你可以在do while循环的开头添加一行代码来清除输入缓冲区中的换行符,如下所示:
  1. do
  2. {
  3.     scanf("%c", &ch); // 读取单个字符,包括换行符
  4.     if (ch != 'Y' && ch != 'N')
  5.     {
  6.         printf("请输入Y或N。\n");
  7.     }
  8. } while (ch != 'Y' && ch != 'N');
复制代码

这样可以确保每次循环只读取一个有效的字符,避免重复打印提示信息。

球一个最佳答案谢谢啦!这对我非常重要!
回答可能有误,多多谅解!
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-9-24 15:58:28 | 显示全部楼层    本楼为最佳答案   
根据你提供的代码,我看到了问题所在。在你的代码中,getchar()函数会读取输入缓冲区中的空白字符(包括回车、换行符等),而不是等待用户输入。这就导致了在执行 do-while 循环时,它会首先读取之前输入的换行符,然后再读取用户的实际输入。

解决这个问题的方法是在 getchar()之前加一个额外的 getchar() 来消耗掉之前的换行符。修改如下:

  1. printf("请问是否需要录入图书信息(Y/N):");
  2. do {
  3.     getchar(); // 消耗之前的换行符
  4.     ch = getchar();
  5.     if (ch != 'Y' && ch != 'N') {
  6.         printf("请输入Y或N。\n");
  7.     }
  8. } while (ch != 'Y' && ch != 'N');
复制代码


这样修改后,程序会等待用户输入 Y 或 N,并且只会打印一次 "请输入Y或N。" 的提示。
如果问题已经解决,请设置最佳答案
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-9-24 16:12:39 | 显示全部楼层

改成scanf("%c", &ch); 还是打印了两次提示
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-5-20 06:03

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表