目录
字符函数和字符串函数
函数介绍
strlen
strcpy
strcat
strcmp
strncpy
strncat
strncmp
strstr
strtok
strerror
memcpy
memmove
memcmp
本章重点
前言
C 语言中对字符和字符串的处理很是频繁,但是 C 语言本身是没有字符串类型的,字符串通常放在 常量字符串 中 或者 字符数组 中。 字符串常量 适用于那些对它不做修改的字符串数。
size_t strlen ( const char * str );
注:size_t 即无符号整型(unsigned int)
strlen使用:
#include
#include
int main()
{
char arr1[] = "abcdef";
char arr2[] = { 'a','b','c','d','e','f','\0'};//计算字符数组长度时末尾必须加'\0',
//否则计算出的长度为随机值.
printf("%d\n", strlen(arr1));
printf("%d\n", strlen(arr2));
return 0;
}
strlen模拟实现:
//方法一:使用计数器
size_t my_strlen(char* str)
{
assert(str);//检查指针有效性
int count = 0;//计数器
//while (*str != '\0')
//{
//count++;
//str++;
//}
//简化
while(*str++)
{
count++;
}
return count;
}
//方法二:递归
size_t my_strlen2(char* str)
{
assert(str);
if (!*str)
{
return 0;
}
else
{
return 1 + my_strlen(str + 1);
}
}
//方法三:指针减指针
size_t my_strlen3(char* str)
{
assert(str);
char* cur = str;
while (*cur)
{
cur++;
}
return cur - str;//两指针相减,结果为他们之间的元素个数
}
注:下面的代码结果如何?
#include
#include
int main()
{
const char*str1 = "abcdef";
const char*str2 = "bbb";
if(strlen(str2)-strlen(str1)>0)//由于strlen函数的返回值为无符号整型,
//所以在计算时,会恒为正数。自己实现时返回值可根据实际情况而定
{
printf("str2>str1\n");
}
else
{
printf("srt1>str2\n");
}
return 0;
}
char* strcpy(char * destination, const char * source );
strcpy使用:
#include
#include
int main()
{
char arr1[10] = "xxxxxxxxx";
char arr2[] = "abcdef";
printf("%s\n", strcpy(arr1, arr2));//将arr2字符串中的内容拷贝到arr1中(包括'\0)
return 0; //需保证arr1的空间大于等于arr2的空间
}
strcpy模拟实现:
#include//用assert函数需包含此头文件
char* my_strcpy(char* dest, const char* src)//返回值为目标空间的起始地址,src中的字符串
//不需要改变,为避免被修改所以再它前面加上const
{
assert(dest && src);//检查指针的有效性
char* ret = dest;//保存目标空间的起始地址,dest后面会移动
//while (*src!='\0')
//{
// *dest = *src;
// dest++;
// src++;
//}
//简化
while (*dest++ = *src++)//src先将值赋给dest,然后dest和src才++
{
;
}
return ret;
}
char * strcat ( char * destination, const char * source );
strcat使用:
#include
#include
int main()
{
char arr1[10] = "abcd";
char arr2[] = "efgh";
printf("%s", strcat(arr1, arr2));//将arr2中的字符串追加到arr1中的字符串后面
return 0; //需保证arr1的空间容纳连接后的字符串
}
strcat模拟实现:
#include
char* my_strcat(char* dest, const char* src)
{
assert(dest && src);
char* ret = dest;
while (*dest)//因为是将src中的字符串追加到dest的后面所以需先找到dest中'\0'的位置
{
dest++;
}
while (*dest++ = *src++)//同strcpy
{
;
}
return ret;
}
其中大概步骤如图:
int strcmp ( const char * str1, const char * str2 );
strcmp使用 :
#include
#include
int main()
{
char arr1[] = "abcdef";
char arr2[] = "abcdxx";
int ret = strcmp(arr1, arr2);
if (ret > 0)
{
printf("arr1 > arr2\n");
}
else if (ret < 0)
{
printf("arr1 < arr2\n");
}
else
{
printf("arr1 = arr2\n");
}
return 0;
}
strcmp模拟实现
#include
int my_strcmp(const char* str1, const char* str2)//str1和str2都不需要被改变
{
assert(str1 && str2);
while (*str1 == *str2)//如果找到不相等的字符直接返回他们的ascll码值之差,
//而不是将所有的字符全部比较完之后再返回.
{
if (*str1 == '\0')
{
return 0;
}
str1++;
str2++;
}
return *str1 - *str2;
}
前面学习了这些字符串函数之后,大家是否有些许收获呢?
其实这些函数中有些是不安全的,比如:strcpy中如果目标空间的大小不能够容纳原空间的字符串,就会造成数组越界访问,strcat中也是同样的道理。还有如果字符串自己给自己追加,是不能使用strcat函数的;如果我们只想比较字符串中的部分字符串的大小,而不是全部,也不能使用strcmp。C语言中给了一些长度受限的字符串函数,而前面的函数是长度不受限的字符串函数。
char * strncpy ( char * destination, const char * source, size_t num );
strncpy使用:
#include
#include
int main()
{
char arr1[] = "xxxxxxxxxx";
char arr2[] = "xx";
char arr3[] = "abcdef";
printf("%s\n", strncpy(arr1, arr3, 8));//拷贝了8个字符,所以字符串中的'\0'也会被拷贝
printf("%s\n", strncpy(arr2, arr3, 1));//拷贝了1个字符,'\0'不会被拷贝
return 0;
}
strncpy模拟实现:
#include
char* my_strncpy(char* dest, const char* src, int count)//参数count为需要拷贝的字节数
{
assert(dest && src);
char* ret = dest;
while(count--)
{
*dest++ = *src++;
}
return ret;
}
char * strncat ( char * destination, const char * source, size_t num );
strncat使用:
/* strncat example */
#include
#include
int main ()
{
char str1[20];
char str2[20];
strcpy (str1,"To be ");
strcpy (str2,"or not to be");
strncat (str1, str2, 6);
puts (str1);
return 0;
}
strncat模拟实现:
char* my_strncat(char* dest, const char* src, int count)
{
assert(dest && src);
char* ret = dest;
while (*dest)//找到目标字符串中'\0'的位置
{
dest++;
}
while (count--)
{
*dest++ = *src++;//从'\0'处开始追加
}
*dest = '\0';//末尾需'\0'
return ret;
}
int strncmp ( const char * str1, const char * str2, size_t num );
strncmp使用:
/* strncmp example */
#include
#include
int main ()
{
char str[][5] = { "R2D2" , "C3PO" , "R2A6" };
int n;
puts ("Looking for R2 astromech droids...");
for (n=0 ; n<3 ; n++)
if (strncmp (str[n],"R2xx",2) == 0)//比较两个字符串的前两个字符
{
printf ("found %s\n",str[n]);
}
return 0;
}
strncmp模拟实现:
int my_strncmp(char* str1, char* str2, int count)
{
assert(str1 && str2);
while (*str1 == *str2)
{
if (count <= 0)
{
return 0;
}
str1++;
str2++;
}
return *str1 - *str2;
}
char * strstr ( const char *str1, const char *str2 );
strstr使用:
/* strstr example */
#include
#include
int main ()
{
char str[] ="This is a simple string";
char * pch;
pch = strstr (str,"simple");
strncpy (pch,"sample",6);
puts (str);
return 0;
}
strstr模拟实现:
char* my_strstr(const char* str1, const char* str2)
{
assert(str1 && str2);
while (*str1)
{
char* cur1 = str1;
char* cur2 = str2;
while (*cur1 == *cur2)
{
cur1++;
cur2++;
if (*cur2 == '\0')
{
return str1;
}
}
str1++;
}
return NULL;
}
此情况下可直接判断str2是否是str1的子串,并且返回相应的值。
当比较到第三个字符时,两字符不相等,而str1并未结束,后续还需比较,所以cur2需回到起始地址,而cur1会回到第一个相等字符的下一个字符的地址处(即第二个b的地址) 。
char * strtok ( char * str, const char * sep );
strtok使用:
/* strtok example */
#include
#include
int main ()
{
char str[] ="- This, a sample string.";
char * pch;
printf ("Splitting string \"%s\" into tokens:\n",str);
pch = strtok (str," ,.-");
while (pch != NULL)
{
printf ("%s\n",pch);
pch = strtok (NULL, " ,.-");
}
return 0;
}
char * strerror ( int errnum );
返回错误码,所对应的错误信息。
/* strerror example : error list */
#include
#include
#include //必须包含的头文件
int main ()
{
FILE * pFile;
pFile = fopen ("unexist.ent","r");
if (pFile == NULL)
printf ("Error opening file unexist.ent: %s\n",strerror(errno));
//出现错误时,会将错误码(一种错误对应一个错误码)放进errno中,strerror会将这个错误码对应的内容打印出来
//errno: Last error number
return 0;
}
例如:
#include
#include
#include
int main()
{
printf("%s\n", strerror(0));
printf("%s\n", strerror(1));
printf("%s\n", strerror(2));
printf("%s\n", strerror(3));
return 0;
}
对应错误码结果为:
void * memcpy ( void * destination, const void * source, size_t num );
memcpy使用:
#include
#include
int main()
{
int arr1[] = { 1,2,3,4,5,6,7,8,9 };
int arr2[10] = { 0 };
int* ret = memcpy(arr2, arr1, 16);
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ", ret[i]);
}
return 0;
}
memcpy模拟实现:
#include
void* my_memcpy(void* dest, const void* src, size_t count)//以字节为单位拷贝
{
assert(dest && src);
void* ret = dest;
while (count--)
{
*(char*)dest = *(char*)src;
dest = (char*)dest + 1;
src = (char*)src + 1;
}
return ret;
}
void * memmove ( void * destination, const void * source, size_t num );
memmove使用:
/* memmove example */
#include
#include
int main ()
{
char str[] = "memmove can be very useful......";
memmove (str+20,str+15,11);
puts (str);
return 0;
}
memmove模拟实现:
情况一:不存在内存重叠,可直接将内容移动。
情况二:存在内存重叠,且目标空间地址大于源空间地址, 如图将1移动到3的位置之后,3就会被覆盖,在移动3时实际移动的是1.
解决方法:从后往前移动,先移动4,再移动3直到所有字节移动完。
情况三:存在内存重叠,且目标空间地址小于源空间地址,如果再从后向前移动也会导致前面的内容被覆盖。
解决方法:从前往后移动,先移动3,再移动4直到所有字节移动完。
实现代码:
void* my_memmove(void* dest, const void* src, size_t count)
{
assert(dest && src);
void* ret = dest;
if (dest > src)//情况二
{
void* cur1 = (char*)src + count - 1;//从后往前移动,需先找到两个空间的末尾地址
void* cur2 = (char*)dest + count - 1;//减1是因为加count之后会直接跳到目标空间和源空间的尾地址的下一个字节的地址
while (count--)
{
*(char*)cur2 = *(char*)cur1;//void*内型指针不能直接加减或解引用操作,需强制类型转换
cur2 = (char*)cur2 - 1;//指针移动
cur1 = (char*)cur1 - 1;
}
}
else//情况1一,三
{
void* cur1 = (char*)src;
void* cur2 = (char*)dest;
while (count--)
{
*(char*)cur2 = *(char*)cur1;
cur2 = (char*)cur2 + 1;
cur1 = (char*)cur1 + 1;
}
}
return ret;
}
int memcmp ( const void * ptr1,
const void * ptr2,
size_t num );
memcmp使用:
/* memcmp example */
#include
#include
int main ()
{
char buffer1[] = "DWgaOtP12df0";
char buffer2[] = "DWGAOTP12DF0";
int n;
n=memcmp ( buffer1, buffer2, sizeof(buffer1) );
if (n>0) printf ("'%s' is greater than '%s'.\n",buffer1,buffer2);
else if (n<0) printf ("'%s' is less than '%s'.\n",buffer1,buffer2);
else printf ("'%s' is the same as '%s'.\n",buffer1,buffer2);
return 0;
}
memcmp模拟实现:
int my_memcmp(const void* str1, const void* str2, int count)
{
assert(str1 && str2);
while (*(char*)str1 == *(char*)str2)
{
count--;
if (count <= 0)
{
return 0;
}
str1 = (char*)str1 + 1;
str2 = (char*)str2 + 1;
}
return (char*)str1 - (char*)str2;
}