例题:分别用strncmp()和memcmp()比较两个数组,返回值是多少?
char s1[4] = {0x10, 0x20, 0x00, 0x30};
char s2[4] = {0x10, 0x20, 0x00, 0x40};
在比较之前我们先了解一下两个函数的具体用法
要说明的一点是,两个函数比较的都是其对应的ASCII码的值。
int strncmp ( const char * str1, const char * str2, size_t n );
功能是把字符串 str1 和 str2 进行比较,最多比较前 n 个字符。
若str1与str2的前n个字符相同,则返回0;
若str1[i] > str2[i],则返回一个大于0的值;
若str1[i] < str2[i],则返回一个小于0的值;
int memcmp(const void *str1, const void *str2, size_t n);
其功能是把存储区 str1 和存储区 str2 的前 n 个字节进行比较。
若str1与str2的前n个字节相同,则返回0;
若str1[i] > str2[i],则返回一个大于0的值;
若str1[i] < str2[i],则返回一个小于0的值;
其实在这两个函数中,因为参数都是char类型,所以字节和字符的区别并不明显
接下来我们调用两个函数看一下结果
printf("%d\n", strncmp(s1, s2, 4));
printf("%d\n", memcmp(s1, s2, 4));
在调用之前,我个人认为,两个函数的调用结果应该都是负数,但实际上,得到的结果是strncmp()的返回值为0, memcmp()的返回值为-1。
于是我查阅了strncmp()的源码:
int __cdecl strncmp (const char *s1, const char *s2, size_t n)
{
if(!n)
return 0;
while(--n && *s1 && *s1 == *s2)
{
s1++;
s2++;
}
return (*(unsigned char)s1 - *(unsigned char)s2);
}
问题的关键就在于代码中的while循环,就本题而言,因为比较的是数组,所以在偏移到0x00时,while循环因为与运算到了0,所以直接退出循环,返回了当前数字的ASCII相减值,因为两个值都为0x00,所以返回值为0。
而且,就函数本身而言,用来比较数组的话,如果s1[]中包含数字0,那么只要对比的字符数量超过了0的位置,都会在0这里退出循环。
而memcmp()对比的则是指针指向的内存空间的数据,数据不相同的时候才会结束。