本内容是重点介绍使用字符函数、字符串函数、内存函数的库函数的使用和注意事项。
前言
求字符串长度字符串查找1.strlen:求字符串长度
strlenstrstr
长度不收限制的字符串函数错误信息报告
strcpystrerror
strcat内存操作函数
strcmpmemcpy
长度受限制的字符串函数介绍memmove
strncpymemset
strncatmemcmp
strncmp字符操作
size_tstrlen (constchar*str);注意事项:
- 字符串已'\0'作为结束标志,strlen函数返回的是在字符串中'\0'前面出现的字符个数(不包含\0)
- 参数指向的字符串必须要以'\0' 结束
- 注意函数的返回值是size_t (unsigned int),是无符号的
#include
#includeint main()
{
char arr1[]= "abcdef";
int ret = strlen(arr1);
printf("%d\n", ret);
return 0;
}
模拟实现strlen函数
#includesize_t my_strlen(const char* str)
{
char* st = str;
while (*st++);
return st-str-1;
}
int main()
{
char arr1[] = "abcdef";
int ret = my_strlen(arr1);
printf("%d\n", ret);
}
文章图片
2. strcpy : 字符串拷贝
char*strcpy(char * destination, constchar * source)注意事项:
- 源字符串必须以 '\0' 结束。
- 会将源字符串中的 '\0' 拷贝到目标空间。
- 目标空间必须足够大,以确保能存放源字符串。
- 目标空间必须可变。
#include
#include
int main()
{
char arr1[20] = { 0 };
char arr2[] = "abcdef";
printf("%s\n",strcpy(arr1, arr2));
return 0;
}
模拟实现strcpy:
#include
#include
char* my_strcpy(char* dest, const char* src)
{
assert(dest && src);
char* ret = dest;
while (*dest++ = *src++)
{
;
}
return ret;
}int main()
{
char arr1[20] = { 0 };
char arr2[10] = "abcdef";
printf("%s\n", my_strcpy(arr1,arr2));
return 0;
}
3.strcat :字符串连接(字符串追加)
char * strcat ( char * destination, const char * source)注意事项:
- 源字符串必须以 '\0' 结束。
- 目标空间必须有足够的大,能容纳下源字符串的内容。
- 目标空间必须可修改。
#include
#include
int main()
{
char a1[20] = "abc";
char a2[] = "def";
printf("%s\n", strcat(a1, a2));
return 0;
}
模拟实现strcat:
#include
char* my_strcat(char* dest, const char* src)
{
char* ret = dest;
while (*dest)
{
dest++;
}
while (*dest++ = *src++)
{
;
}
return ret;
}
int main()
{
char a1[20] = "abc";
char a2[] = "def";
printf("%s\n", my_strcat(a1, a2));
}
文章图片
4.strcmp :字符串比较,比较的是字符串的内容,不是长度
int strcmp ( const char * str1,const char * str2);标准规定:
- 第一个字符串大于第二个字符串,则返回大于0的数字
- 第一个字符串等于第二个字符串,则返回0
- 第一个字符串小于第二个字符串,则返回小于0的数字
#include
#include
int main()
{
char a1[] = "abcd";
char a2[] = "abcd";
int ret = strcmp(a1, a2);
printf("%d\n", ret);
return 0;
}
模拟实现strcmp:
#include
int my_strcmp(const char* s1, const char* s2)
{
while (*s1 == *s2)
{
if (*s1 == '\0')
{
return 0;
}
s1++;
s2++;
}
return *s1 - *s2;
}
int main()
{
char a1[] = "abcd";
char a2[] = "abcd";
int ret = my_strcmp(a1, a2);
printf("%d\n", ret);
return 0;
}
5.strncpy
char * strncpy ( char * destination , const char * source , size_t num);注意事项:
- 拷贝num个字符从源字符串到目标空间。
- 如果源字符串的长度小于num,则拷贝完源字符串之后,在目标的后边追加0,直到num个。
#include
#include
int main()
{
char arr1[20] = { 0 };
char arr2[] = "abcdef";
printf("%s\n", strncpy(arr1, arr2 + 2, 2));
//输出结果cd
return 0;
}
6.strcat
char * strncat ( char * destination , const char * source ,size_t num);实例:
#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);
//输出结果,To be or not
return 0;
}
7.strncmp
int strncmp ( const char * str1, const char * str2, size_t num );注意事项: 比较到出现另个字符不一样或者一个字符串结束或者num个字符全部比较完。
文章图片
实例:
#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]);
//输出结果R2D2R2A6
}
return 0;
}
8. strstr :字符串查找函数 在一个字符串中查找子字符串
char * strstr ( const char * str2, const char * str1);实例:
#include
#include
int main()
{
char arr1[] = "i am a good student,hehe student";
char arr2[] = "student";
//查找arr1中arr2第一次出现的位置
char* ret = strstr(arr1, arr2);
if (ret == NULL)
{
printf("找不到\n");
}
else
{
printf("%s\n", ret);
//输出 student,hehe student
}
return 0;
}
模拟实现strstr:
#include
#include
char* my_strstr(const char* str1, const char* str2)
{
assert(str1 && str2);
char* s1;
char* s2;
char* cp = str1;
if (*str2 == '\0')
{
return str1;
}
while (*cp)
{
s1 = cp;
s2 = str2;
while (*s1 != '\0' && *s2 != '\0' && *s1 == *s2)
{
s1++;
s2++;
}
if (*s2 == '\0')
{
return cp;
}
cp++;
}
return NULL;
}int main()
{
char a1[] = "hello my name is wz zz";
char a2[] = "wz";
char* ret = my_strstr(a1, a2);
if (ret == NULL)
{
printf("找不到了\n");
}
else
{
printf("%s\n", ret);
}
return 0;
}
9.strtok :分割字符串
char * strtok ( char * str, const char * sep );注意事项 :
- sep参数是个字符串,定义了用作分隔符的字符集合
- 第一个参数指定一个字符串,它包含了0个或者多个由sep字符串中一个或者多个分隔符分割的标 记。
- strtok函数找到str中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注: strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容 并且可修改。)
- strtok函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串 中的位置。
- strtok函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标 记。
- 如果字符串中不存在更多的标记,则返回 NULL 指针。
#include
#include
int main()
{
char arr1[] = "wzzz@bwwwu.errrh";
char arr2[100] = { 0 };
//临时数据
//zpw\0bitedu.tech\0 char sep[] = "@.";
strcpy(arr2, arr1);
char* ret = NULL;
//分割字符串 for (ret=strtok(arr2, sep);
ret!=NULL;
ret=strtok(NULL, sep))
{
printf("%s\n", ret);
} //strtok(arr2, sep);
//strtok(NULL, sep);
return 0;
}
10.strerror
char * strerror(int errnum);返回误码,所对应的错误信息 。
实例:
#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;
}
文章图片
11.memcpy :拷贝的整形数据
void * memcpy ( void * destination, const void * source, size_t num );注意事项:
- 函数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。
- 这个函数在遇到 '\0' 的时候并不会停下来。
- 如果source和destination有任何的重叠,复制的结果都是未定义的。
#include
#include
struct {
char name[40];
int age;
} person, person_copy;
int main()
{
char myname[] = "Pierre de Fermat";
memcpy(person.name, myname, strlen(myname) + 1);
person.age = 46;
memcpy(&person_copy, &person, sizeof(person));
printf("person_copy: %s, %d \n", person_copy.name, person_copy.age);
return 0;
}
模拟实现memcpy:
#include
#include
void* my_memcpy(void* dest, const void* src, size_t count)
{
void* ret = dest;
assert(dest && src);
while (count--)
{
*(char*)dest = *(char*)src;
dest = (char*)dest+1;
src = https://www.it610.com/article/(char*)src+1;
}
return ret;
}int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
int i = 0;
my_memcpy(arr + 2, arr, 16);
for (i = 0;
i < 10;
i++)
{
printf("%d ", arr[i]);
}
return 0;
}
12.memmove :内存拷贝时,出现内存重复的现象
void * memmove ( void * destination, const void * source, size_t num );注意事项:
- 和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。
- 如果源空间和目标空间出现重叠,就得使用memmove函数处理。
#include
#include
int main ()
{
char str[] = "memmove can be very useful. ........... ";
memmove (str+20,str+15,11);
puts (str);
return 0;
}
模拟实现memmove:
#include
#include
void* my_memmove(void* dest, const void* src, size_t count)
{
void* ret = dest;
assert(dest && src);
if (dest < src)
{
//dest在src左边时,从前往后拷贝
while (count--)
{
*(char*)dest = *(char*)src;
dest = (char*)dest + 1;
src = https://www.it610.com/article/(char*)src + 1;
}
return ret;
}
else
{
//dest在src右边时,从后往前拷贝
while (count--)
{
*((char*)dest+count) = *((char*)src+count);
}
return ret;
}
}int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
int i = 0;
my_memmove(arr + 2, arr, 16);
for (i = 0;
i < 10;
i++)
{
printf("%d ", arr[i]);
}
return 0;
}
13.memcmp
int memcmp ( const void * ptr1, const void * ptr2, size_t num );注意事项:
- 比较从ptr1和ptr2指针开始的num个字节
- 返回值:
文章图片
实例:
#include
#include
int main()
{
int arr1[] = { 1,2,3,4,5 };
int arr2[] = { 1,2,3,6,6 };
int ret = memcmp(arr1, arr2, 12);
printf("%d\n", ret);
//输出结果为 0
return 0;
}
14. memset
void * memset(void *dest,int c,size_t count);实例:
#include
#include
int main()
{
int arr[] = { 1,2,3,4,5,6 };
memset(arr, 1, 20);
return 0;
}
【c语言|C语言中,C进阶中字符函数、字符串函数、内存函数详解。】
推荐阅读
- 笔记|面试官让我写strlen函数(|详解字符串函数与内存函数【C语言/进阶】)
- c语言|C语言进阶学习日志(字符串和内存函数(一))
- C语言篇|【C语言学习】字符函数和字符串函数【进阶详解篇15】
- 算法|【算法】【C语言进阶】C语言字符串操作宝藏级别汇总 strtok函数 strstr函数该怎么用(【超详细的使用解释和模拟实现】)
- C语言编程学习|【C语言进阶学习笔记】三、字符串函数+内存函数详解(2)
- C语言笔记|C语言进阶笔记(六) | 详解内存函数及其模拟
- 排序算法|常见的排序算法(上)
- 数据结构|LeetCode每日一刷 --- 拿捏顺序表经典面试题
- 数据结构|LeetCode每日一刷 --- 手撕单链表习题(1)