在把数组作为参数传递给函数时,可以通过sizeof运算符告诉函数数组的大小吗?
不可以。当把数组作为函数的参数时,你无法在程序运行时通过数组参数本身告诉函数该数组的大小,因为函数的数组参数相当于指向该数组第一个元素的指针。这意味着把数组传递给函数的效率非常高,也意味着程序员必须通过某种机制告诉函数数组参数的大小。
为了告诉函数数组参数的大小,人们通常采用以下两种方法:
第一种方法是将数组和表示数组大小的值一起传递给函数,例如memcpy()函数就是这样做的:
char source[MAX],dest[MAX];
/*... */
memcpy(dest,source,MAX);
第二种方法是引入某种规则来结束一个数组,例如在C语言中字符串总是以ASCII字符NUL('\0')结束,而一个指针数组总是以空指针结束。请看下述函数,它的参数是一个以空指针结束的字符指针数组,这个空指针告诉该函数什么时候停止工作:
void printMany(char *strings口)
{
int i;
i=0;
while(strings[i]!=NULL)
{
puts(strings[i]);
++i;
}
}
正象9.5中所说的那样,C程序员经常用指针来代替数组下标,因此大多数C程序员通常会将上述函数编写得更隐蔽一些:
void printMany(char *strings[])
{
while(*strings)
{
puts(*strings++);
}
}
尽管你不能改变一个数组名的值,但是strings是一个数组参数,相当于一个指针,因此可以对它进行自增运算,并且可以在调用puts()函数时对strings进行自增运算。在上例中,while(*strings)
就相当于
while(*strings !=NULL)
在写函数文档(例如在函数前面加上注释,或者写一份备忘录,或者写一份设计文档)时,写进函数是如何知道数组参数的大小是非常重要的,例如,你可以非常简略地写上“以空指针结束”或“数组elephants中有numElephants个元素”(如果你在程序中用数字13表示数组的大小,你可以写进“数组arr中有13个元素”这样的描述,然而用确切的数字表示数组的大小不是一种好的编程习惯)。
为了告诉函数数组参数的大小,人们通常采用以下两种方法:
第一种方法是将数组和表示数组大小的值一起传递给函数,例如memcpy()函数就是这样做的:
char source[MAX],dest[MAX];
/*... */
memcpy(dest,source,MAX);
第二种方法是引入某种规则来结束一个数组,例如在C语言中字符串总是以ASCII字符NUL('\0')结束,而一个指针数组总是以空指针结束。请看下述函数,它的参数是一个以空指针结束的字符指针数组,这个空指针告诉该函数什么时候停止工作:
void printMany(char *strings口)
{
int i;
i=0;
while(strings[i]!=NULL)
{
puts(strings[i]);
++i;
}
}
正象9.5中所说的那样,C程序员经常用指针来代替数组下标,因此大多数C程序员通常会将上述函数编写得更隐蔽一些:
void printMany(char *strings[])
{
while(*strings)
{
puts(*strings++);
}
}
尽管你不能改变一个数组名的值,但是strings是一个数组参数,相当于一个指针,因此可以对它进行自增运算,并且可以在调用puts()函数时对strings进行自增运算。在上例中,while(*strings)
就相当于
while(*strings !=NULL)
在写函数文档(例如在函数前面加上注释,或者写一份备忘录,或者写一份设计文档)时,写进函数是如何知道数组参数的大小是非常重要的,例如,你可以非常简略地写上“以空指针结束”或“数组elephants中有numElephants个元素”(如果你在程序中用数字13表示数组的大小,你可以写进“数组arr中有13个元素”这样的描述,然而用确切的数字表示数组的大小不是一种好的编程习惯)。