c语言中的一些问题(scanf循环、三维数组做形参)

本文介绍了C语言中多维数组的处理方法及如何有效地使用scanf函数进行输入验证,探讨了不同形式的多维数组传递给函数的方式,并提供了处理scanf函数潜在问题的有效解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

(1)关于多维数组做形参

      数组指针是指向数组的指针,是一个指针,其指向的类型是数组;定义方式为(*p)[][]

      指针数组是元素为指针的数组,是一个数组,其中的元素为指针;   定义方式为 *p[][]

          

      如果要将数组a[][]传递给函数fun(),这个函数的定义中形参需要这样定义 void  fun(int a[][6] )  调用的时候就可以直接使用fun(a)。 在二维数组传递的过程中,第一维的维数是可以为空,三维照样如此~~~~~

      函数调用三维数组,可以使用的形式有如下几种:

*1           int a[4][6][25];

                 void fun(int b[4][6][25])               调用: fun(a);

*2            int a[4][6][25];

               void fun (int (*p)[6][25])              调用:  fun(a);

                  使用:      *(*(*(p+i)+j)+k)            i,j,k三个变量分别是代表数组的三维

*3           void fun (int (*p)[4][6][25])        调用: fun (&a);

                  使用:      *(*(*(*p+i)+j)+k)            i,j,k三个变量分别是代表数组的三维

*4            int a[4][6][25];

                  void fun (int *p)                           调用: fun(&a[0][0][0]);  

       这种方式使用过程中就是将三维数组的首地址付给一个简单的指针变量p, p[600]依次代表着数组a[4][6][25]中的各个数;

      一般数组作为参数传递最好使用指针作为形参,以减少内存的使用。

 

(2)scanf循环

       

很多人在使用scanf的时候都不会去检查它的返回值, 包括我在这之前也没有去注意它, 如果你方便去翻翻c语言的教科书, 估计也很难找到有检查scanf的例子或代码段, 至少我还没有见过, 当然, 书上的代码毕竟只是一个练习, 和真正用于产品级别的代码有很大的差别。

一个偶然的机会, 让我注意到检查scanf返回值的重要性, 写下来和大家分享一下:)

下面是一个最简单不过的scanf用法,但却有隐患:

int n;

scanf( "%d", &n);

...

scanf要求取一个整数,但如果输入的是一个字母,或其它不是数字的字符时,情况会怎么样呢? 结果是scanf不读取数据,不修改n的值直接返回了,这样,n就变成了一个未被初始化的变量,其值不可确定,因为其值不确定,所以就算你对n作是否合法的判断,也还是可能会有问题,

你也许会说,用户应该会输入一个合法的值的,但是,有时scanf并不总是从键盘中读入数据,它可能从其它的设备中去读,例如管道,UNIX中允许这样的玩法。你也许还会说,情况没有那么糟啦,可以先初始化n为零,再去判断n是否合法嘛,例如下面这样:

int n = 0;

scanf( "%d", &n);

if (n != 0)

{

    //OK,读到用户的值了

}

这样好像还是有麻烦啊,这个初始值你怎么定义才好? 0吗? 程序若是需要0的时候怎么办?

让我们再看下面的代码,要求用户输入一个大于0的数为止:

int n;

for(;;) {

    printf( "pls input:");

    scanf( "%d", &n);

    if (n > 0) {

        break;

    }

}

上面的代码中,当用户输入的不是一个整数时,会死循环,而不是用户想要的:当用户输入非法值时一直让用户输入一个合法值为止。为什么会这样呢,因为当scanf检查到所输入的是不合法的值时,它并不会读取它,因此数据会留在缓冲区中,当下一次调用scanf时,会去读缓冲区中的数据,因此仍然会读到一个非法的值

----------------------------------------------------------------------------------------------

以下是比较好的解决方案:

方案1:

1. 用类似 iReadRslt = scanf( "%8d", &iMaxDataNum ); 的方法取得scanf的返回值

并对其进行是否为0的判断,可以检查输入的合法性(比如要求输入整数,对方却输入字母,此时返回值为0)。

2. 上面引文中提到的输入值不合法,同时不合法的输入值贮留输入缓冲区导致scanf不断读取非法值,程序死循环的问题,可以使用下面方法解决:

while( ( 0 >= iMaxDataNum ) || ( 0 >= iReadRslt ) )

{

if( 0 >= iReadRslt )

{

   while( '/n' != getchar() )

   {

    continue;

   }

}

printf( "Invalid number, please input the correct one." );

printf( "/nHow many Tel No. can be saved!/n" );

iReadRslt = scanf( "%8d", &iMaxDataNum );

}

其中while条件中的0 >= iMaxDataNum 是在判断输入值的范围问题,不在我们讨论范围。

后面的

if( 0 >= iReadRslt )

{

   while( '/n' != getchar() )

   {

    continue;

   }

}

就是用来解决问题的。当用户输入非法值时,用一个while循环,将本次输入的非法值连同最后确定输入时敲的回车,都从输入缓冲中读出来。这样就避免了上次的非法输入值被不断读入,从而造成的多次循环

----------------------------------------------------------------------------------------------------------------------

方案2:

直接使用缓冲区清除函数fflush(stdin); //stdin代表标准输入;

以下是一个例子:

/* 程序功能:解决scanf在非法输入后的死循环问题 */

#include <stdio.h>

#include <stdlib.h>

void main()

{

int func = -1;   //初始化为非法标志-1;

while(1)

{

   printf ( "-----------------");

   printf ( "/n*** 主 菜 单 ***/n" );

   printf ( " 0. 退出/n");

   printf ( " 1. show hello/n");

   printf ( " 2. 清楚屏幕/n");

   printf ( "-----------------/n");

   printf("选择(0--2):");

   scanf("%d",&func);

   if(getchar()!='/n')func=-1; //避免类似“1qweqweqw”这样的输入,即只能输入0或1或2;

   fflush(stdin); //清空输入缓冲区;

   switch ( func )

   {

    case 0:

     exit(0);

     break;

    case 1:

     printf ( "/nhello,world~~~/n/n");

     break;

    case 2:

     system("cls");

     break;

    default:

     printf("输入错误!/n");

     break;

   } /* end switch */

   func = -1;

}

}

 

 

(3)循环scanf()可能导致直接读入' /n ',无法正确读入键入值

#i nclude <stdio.h>
int main()
{
int a;
char c; do
{
       scanf("%d",&a);
       scanf("%c",&c);
       printf("a=%d     c=%c/n",a,c);
       /*printf("c=%d/n",c);*/
}while(c!='N');
} scanf("%c",&c);
这句不能正常接收字符,什么原因呢?

我们用printf("c=%d/n",c);将C用int表示出来,启用 printf("c=%d/n",c);这一句,看看scanf()函数赋给C到底是什么,结果是 c=10 ,ASCII值为10是什么?换行即/n.对了,我们每击打一下"Enter"键,向键盘缓冲区发去一个“回车”(/r),一个“换行"(/n),在这里 /r被scanf()函数处理掉了(姑且这么认为吧^_^),而/n被scanf()函数“错误”地赋给了c.

也就是说:scanf()和getchar()函数是从输入流缓冲区 中读取值的,而并非从键盘(也就是终端)缓冲区读取。而读取时遇到回车(/n)而结束的,这个/n会一起读入输入流缓冲区的,所以第一次接受输入时取走字 符后会留下字符/n,这样第二次的读入函数直接从缓冲区中把/n取走了,显然读取成功了,所以不会再从终端读取!因此后面只要加上fflush(stdin)函数释放缓冲区数据就可以了.

解决办法:可以在两个scanf()函数之 后加个fflush(stdin);,还有加getch(); getchar();也可以,但是要视具体scanf()语句加那个,这里就不分析了,读者自己去摸索吧。但是加fflush(stdin);不管什么情 况都可行。

函数名: fflush
功 能: 清除一个流
用 法: int fflush(FILE *stream);#i nclude <stdio.h>
int main()
{
int a;
char c; do
{
       scanf("%d",&a);
       fflush(stdin);
       scanf("%c",&c);
       fflush(stdin);
       printf("a=%d     c=%c/n",a,c);
    }while(c!='N');
}   

这里再给一个用“空格符”来处理缓冲区残余信息的示例:

运行出错的程序:
#i nclude <stdio.h>
int main()
{
int i;
char j;
for(i = 0;i < 10;i++)
{
       scanf("%c",&j);/*这里%前没有空格*/
}
}

使用了空格控制符后:

#i nclude <stdio.h>
int main()
{
int i;
char j;
for(i = 0;i < 10;i++)
{
       scanf(" %c",&j);/*注意这里%前有个空格*/
}
}  

         

<think>好的,我现在需要回答用户关于如何在C语言中将二维字符数组作为函数形参问题。首先,我得回忆一下C语言数组作为函数参数的相关知识。 用户提到的是二维字符数组,所以我要确定二维数组作为参数传递的正确语法。在C语言中,数组作为函数参数时,通常需要传递数组的指针。对于二维数组,列的大小必须明确指定,因为编译器需要知道每一行的长度来计算内存地址。例如,一个二维字符数组char arr[3][10],在函数声明时应该写成char arr[][10],或者使用指针形式char (*arr)[10]。 接下来,我需要考虑用户提供的引用内容。引用[1]和[2]提到了二维字符数组的初始化和相关函数的使用,比如printf和scanf的处理。特别是引用[2]中的代码示例,展示了函数定义和调用的方式,这可能对回答有帮助。 可能用户会混淆二维数组和指针数组的区别。例如,如果数组是动态分配的或者每行的长度不同,可能需要使用指针数组(如char *arr[]),但这里用户的问题明确是二维字符数组,所以应保持列的大小固定。 另外,需要注意函数声明中是否需要指定行数。实际上,行数可以省略,但列数必须指定。比如void func(char arr[][10], int rows)是正确的,而void func(char arr[][], int rows)则会报错,因为列数未知。 还需要提醒用户,传递二维数组时,通常还需要传递行数作为参数,因为数组在函数内部无法自行确定行数。这在引用[2]的示例中,主函数调用add函数时传递了a和b的值,类似地,这里也需要传递行数。 可能的错误写法包括未指定列的大小,或者错误地使用双重指针(char **arr),这适用于指针数组而非二维数组。用户需要注意区分这两种情况。 总结步骤: 1. 函数形参声明为二维数组,指定列的大小,例如char arr[][10]。 2. 或者使用数组指针的形式,例如char (*arr)[10]。 3. 函数调用时直接传递数组名,因为数组名会退化为指向第一个元素的指针。 4. 在函数内部使用数组时,注意列数已确定,行数可能需要作为另一个参数传递。 需要验证这些知识是否正确,比如查看C语言标准或可靠的资料。根据之前的C语言经验,这些是正确的法。同时,引用[1]中的示例使用了二维字符数组的输入输出,但没有涉及函数参数,所以需要结合其他知识。 最后,给出一个示例代码,展示如何定义函数和传递二维字符数组参数,这样用户能更直观地理解。例如: void printStrings(char arr[][10], int rows) { for (int i = 0; i < rows; i++) { printf("%s\n", arr[i]); } } 这样,用户可以看到函数如何接收二维数组,并使用行数进行遍历。</think>在C语言中,将二维字符数组作为函数形参时,需明确数组的列维度以支持内存寻址。以下是具体实现方法及注意事项: ### 一、二维字符数组形参定义方法 1. **直接声明列维度** 函数形参需固定列维度,例如处理每行最多10字符的二维数组: ```c void func(char arr[][10], int rows) { // 通过arr[i][j]访问元素 } ``` *适用场景*:已知列宽且固定的数据存储,如密码表、固定长度字符串集合[^1]。 2. **使用数组指针形式** 通过指针语法声明列维度: ```c void func(char (*arr)[10], int rows) { // 等价于第一种写法 } ``` *优势*:更直观体现指针特性,适用于需要指针运算的场景。 ### 二、函数调用示例 ```c #include <stdio.h> // 函数声明 void print_array(char words[][20], int size); int main() { char dict[3][20] = {"Hello", "World", "C_Programming"}; print_array(dict, 3); // 直接传递数组名 return 0; } // 函数定义 void print_array(char words[][20], int size) { for(int i=0; i<size; i++){ printf("%s\n", words[i]); // 输出每行字符串[^1] } } ``` ### 三、易错点及注意事项 1. **列维度必须显式声明** 若省略列维度如`char arr[][]`将导致编译错误,编译器无法计算元素偏移量。 2. **行数参数需额外传递** C语言不会自动传递数组尺寸,需通过`int rows`参数明确行数,如引用[2]中`add`函数显式接收操作数的设计思路。 3. **与指针数组的区别** 若实际使用`char *arr[]`(指针数组),形参应声明为`char **arr`,这与二维数组的内存结构完全不同。 ### 四、扩展应用 当处理变长字符串时,推荐使用指针数组+动态内存分配: ```c void process_strings(char **strings, int rows) { // 处理每个strings[i] } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值