C/C++函数传递参数需要指定缓冲区大小
提醒:本页面将不再更新、维护或者支持,文章、评论所叙述内容存在时效性,涉及技术细节或者软件使用方面不保证能够完全有效可操作,请谨慎参考!
原文发表于2009年2月5日 标题是《一个自己犯的C/C++错误》
以前自己写程序时经常犯的错误,后来才开始重视起来,为了更好的说明这个错误,我将演示代码贴出来:
#include <stdio.h>
#include <string.h>
#include <memory.h>
// 写出缓冲区的16进制值
void print_byte(void *ptr, size_t size)
{
unsigned i;
unsigned char *bptr = (unsigned char *)ptr;
for(i = 0; i < size; i++) {
printf("%x ", *(bptr + i));
}
putchar('\n');
}
// 测试函数1
void func_test(char *pbuf, size_t size)
{
printf("SIZE=%d ? \n", sizeof(pbuf));
memset(pbuf, 0, sizeof(pbuf)); // 我以前经常犯的错误
print_byte(pbuf, size);
}
// 测试函数2
void func_test1(char *pbuf, size_t size)
{
memset(pbuf, 0, size);
print_byte(pbuf, size);
}
int main(void)
{
char buffer[] = "0123456789"; // 字符缓冲区
printf("SIZE=%d\n", sizeof(buffer));
print_byte(buffer, sizeof(buffer));
memset(buffer, 0, sizeof(buffer));
print_byte(buffer, sizeof(buffer));
strcpy(buffer, "0123456789");
func_test(buffer, sizeof(buffer));
func_test1(buffer, sizeof(buffer));
return 0;
}
可以看出,表面上这个错误是关于在函数内部将传参指向的缓冲区清零的,调试上述程序后发现主函数里定义的缓冲区被全部成功设置为0,而将这个缓冲区地址传入函数func_test后只有前4个字节被置0,那么问题出在哪里呢?问题就在sizeof上,在主函数上sizeof算得缓冲区为11(包含字符串结尾\0),而函数func_test里算得是4,很明显只是计算的指针的大小。
疑惑就在这里,buffer是数组名不就是地址吗,为什么传参后sizeof值就不算整个数组的大小而只算指针的大小呢?
其实这个问题很容易buffer是数组名,sizeof(数组名)算得的是整个数组占用的字节数,一旦赋值给任何指针(函数传参也相当于一种赋值),也就算的是这个指针的占用空间,和数组就没任何关系了。若还是算的事数组占用空间,那这个指针的占用就没办法计算了。
理解了这些就不难理解一些SDK提供的接口为什么要你传入缓冲区地址后还要你给出大小了,就像上面的修正函数func_test1。
通过这件事我知道什么事都不能想当然。不过对我过去的代码影响不是很大的因为在Windows API下经常使用固定大小MAX_PATH。于是大小就这样算了sizeof(TCHAR) * MAX_PATH。