❤️ 作者简介 :RO-BERRY 致力于C、C++、数据结构、TCP/IP、数据库等等一系列知识,对纯音乐有独特的喜爱
日后方向 : 偏向于CPP开发以及大数据方向,如果你也感兴趣的话欢迎关注博主,期待更新
C语言中对字符和字符串的处理很是频繁,但是C语言本身是没有字符串类型的,字符串通常放在常量字符串中或者字符数组中。
字符串常量适用于那些对它不做修改的字符串函数
在这里相信你知道了字符以及字符串的概念,那你是否知道C语言中库里有定义好的字符函数和字符串函数呢?
如果你之前做过了解,但你是否用过呢?
今天我们一起来学习一下我们C语言中字符函数以及字符串函数的了解以及使用
很多情况下,我们除了存储定义一个字符串,我们还需要做很多其他的工作。可以对字符串进行分析、合并、拷贝、修改等等其他操作。那我们怎么样进行实现呢?
我们能想到的,C语言开发的程序员们肯定也会想到这些麻烦,所以C语言库中所包含的字符串函数就是用来帮助我们轻松实现实现这些操作的
在C语言中字符串函数都包含在
我们定义了一个字符串,我们想要求其长度,我们应该怎么做呢?
还记得我们以前说过,字符串存在字符数组里,我们想要求字符串长度我们可以用一个sizeof()函数来求,可能有的小伙伴还不知道,我们再来重新看一段代码了解sizeof()函数的使用:
#include
int main()
{
char arr[] = "abcde";
int sz = sizeof(arr) / sizeof(arr[0]);
printf("%d", sz);
}
下面是该程序的输出
老手已经在点头了,新手还在问为什么
那为什么这里输出的是6呢,明明是5个字符。
其实我们在这样存储字符数组的时候,后面是有一个\0的,所以才会返回数字6
我们来看arr数组中到底存了些什么
回归正题,我们想求字符的长度,如果每次都这样是不是很麻烦?
接下来,我们来看看strlen函数是如何帮我们解决麻烦的
strlen函数为我们提供了一个求字符串长度很方便的工具,它的格式为:
size_t strlen ( const char * str );
1.函数参数为const char *str
我们传入的str字符串就是求长度的目标字符串,而且前面加了const,表明这是不可修改的字符串,这是为了防止原数据的损坏
2.返回参数为size_t
返回值的类型为size_t,size_t 类型表示C中任何对象所能达到的最大长度,它是无符号整数。
我们来看一段代码:
int main()
{
if (strlen("abc") - strlen("abcde")>0)
printf("大于");
else
printf("小于等于");
return 0;
}
我们口算一下,strlen(“abc”)和strlen(“abcde”)的结果,既然是无符号整形,那就是3和5;我们一减是不是就是-2,那我们应该输出小于等于对吗?
我们来看输出结果:
程序告诉我们这个程序输出结果为大于
这是为什么?
我们要先知道,无符号整形的取值为0~4294967295(这里不懂的请转到数据在内存中的存储)
我们定义一个unsigned a去接收返回值,来看一看a的值为多少
我们会发现a的值为4294967294
a的值是大于0的,所以程序输出大于
正因为如此,我们在使用strlen函数来比较大小的时候,尽量少使用加减法,直接用大鱼小于号来做比较就可以,如果硬是要用加减法,我们最好将其类型强转为int,方可做大小比较
这样我们就可以得到我们所需要的答案
在这里我们就是提一下,这是一个易错点,需要注意哦
好了我们再次回归正题,我们大概能了解strlen函数的作用了,我们再来看strlen与sizeof算出来的答案有没有出入
我们会发现,strlen是不会将 ’\0’ 算进去的,并且我们字符串里面的 ‘\0’ 为strlen的终止字符,从这里开始,我们的strlen就会停止往下计算长度
如果你嫌很难记住的话
我们可以把strlen分解来看:
合起来就是字符串长度函数,这样子就方便记很多啦
模拟实现strlen我们可以有三种方法:
int my_strlen(const char * str)
{
int count = 0;
while(*str)
{
count++;
str++;
}
return count;
}
int my_strlen(const char * str)
{
if(*str == '\0')
return 0;
else
return 1+my_strlen(str+1);
}
int my_strlen(char *s)
{
char *p = s;
while(*p != ‘\0’ )
p++;
return p-s;
}
综上:
我们需要格外注意
- 字符串已经 ‘\0’ 作为结束标志,strlen函数返回的是在字符串中 ‘\0’ 前面出现的字符个数(不包含 ‘\0’ )。
- 参数指向的字符串必须要以 ‘\0’ 结束。
- 注意函数的返回值为size_t,是无符号的( 易错 )
上面我们实现了如何使用strlen来快速求字符串的长度,那我们想实现字符串的拷贝又怎么办呢?
我们如果想要将一个字符串的内容拷贝到另一个字符串里去,我们就只能使用for循环嵌套,一个一个拷贝上去,那我们可不可以简便一点呢?
说到这里,通过我的话你也知道了strcpy的作用是什么
没错,就是用来做字符串的拷贝
strcpy是C语言给我们提供方便进行字符串拷贝的,其格式为:
char* strcpy(char * destination,const char * source );
1.返回类型为char*
char*指向的是我们已经拷贝完毕的数组的首元素地址
2.函数参数类型为char*和const char*
一个可变一个不可变,不可变的字符串是我们需要进行拷贝的,拷贝到可变数组里,也就是英文单词source(来源)和destination(目的地),我们将来自source字符串的数据拷贝到destination字符串里,
destination字符串必须为可变的
我们来看代码:
int main()
{
char arr1[] = "xxxxxxxxxxx";
char arr2[] = "abcdef";
strcpy(arr1, arr2);
puts(arr1);
return 0;
}
程序输出结果:
我们会看到我们输出只输出了abcdef
理论上我们应该输出的是abcdefxxxx
这是为什么呢?
我们来看arr1里面的存储情况
♂️我们会发现,我们这个函数会将 ‘\0’ 也一起拷贝过去,而我们的printf由于输出类型为%s,所以在看到\0,会终止,将\0之前的看作为字符串输出出来。
♀️同时,我们也可以看到,我们strcpy的终止条件为看到了\0,这也表明了我们源字符串必须要以 ‘\0’ 结束。
如果我们的目的字符串空间不够大会出现什么情况呢?
我们再来看一个例子:
int main()
{
char arr1[] = "xxxx";
char arr2[] = "abcdef";
strcpy(arr1, arr2);
printf("%s", arr1);
return 0;
}
我们发现输出的时候出现了异常
⚡这说明我们的目标字符串必须为足够大的。
我们同样可以把strcpy分解来看:
合起来就是字符串拷贝函数,这样子就方便记很多啦
char *my_strcpy(char *dest, const char*src)
{
char *ret = dest;
assert(dest != NULL);
assert(src != NULL);
while((*dest++ = *src++))
{
;
}
return ret;
}
assert()是一个断言函数,用于测试两个值是否相等。当这两个值不相等时,会抛出一个错误。它常用于单元测试中,用来验证代码的正确性。
总结:
- 源字符串必须以 ‘\0’ 结束。
- 会将源字符串中的 ‘\0’ 拷贝到目标空间。
- 目标空间必须足够大,以确保能存放源字符串。
- 目标空间必须可变
我们定义了一个字符串,我们因为疏忽,尾部弄掉了另外几个字符,我们想追加字符怎么办?
strcat使用来我们追加字符串的,我们可以使用它来将源字符串追加到目的字符串
char * strcat ( char * destination, const char * source );
1.返回类型为char*,这是我们进行了追加了字符串之后的字符串
2.函数参数为char * destination,const char * source,我们将source字符串追加到destination字符串中进行返回,追加的字符串为不可改变
我们依旧来看代码展示:
int main()
{
char arr1[] = "xxxx";
char arr2[] = "abcdef";
strcat(arr1, arr2);
printf("%s", arr1);
return 0;
}
输出结果
⚡我们发现出现了异常,此异常为越界访问
说明我们arr1的空间需要足够大
我们将其加长空间再来试一次:
int main()
{
char arr1[20] = "xxxxx";
char arr2[] = "abcdef";
strcat(arr1, arr2);
printf("%s", arr1);
return 0;
}
这个时候我们可以看到arr2的内容已经拷贝到了arr1里面
那我们strcat又是如何判断结束的呢?
难道还是以\0来结束判断的吗
我们再来看一段代码:
正所谓我们程序员熟悉任何代码的操作就是不停的敲代码,我们只有在敲代码的时候不断修改调试,才能知道我们的熟练水平。所以趁现在,我们多敲多看
回归正题,我们来看演示代码:
int main()
{
char arr1[20] = "xxxxx";
char arr2[] = {'a','b','c','d','e','f'};
strcat(arr1, arr2);
printf("%s", arr1);
return 0;
}
为什么乱码了, 一乱码就出现烫烫烫,我们的烫哥真是无处不在。
我们在arr2中采取这种方式存储字符串是没有\0的,所以这也说明了我们的strcat函数确实是以\0来判断停止条件的
有小伙伴还提出来一个问题:
如果我们字符串自己给自己追加会怎么样?
我们不知道,我们就敲出来,计算机是不会骗我们的
看代码:
int main()
{
char arr[] = "abcdef";
strcat(arr, arr);
printf("%s", arr);
return 0;
}
输出结果:
我们可以看到,又出现了异常,这是为什么呢?
我们可以从参数列表来看,目标字符串为可变的,而源字符串为不可变的,如果传入俩一样的,怎么可能即是可变又是不可变的呢?自相矛盾了
我们来把strcat分解来看:
合起来就是字符串追加函数,这样子就方便记很多啦
char *my_strcat(char *dest, const char*src)
{
char *ret = dest;
assert(dest != NULL);
assert(src != NULL);
while(*dest)
{
dest++;
}
while((*dest++ = *src++))
{
;
}
return ret;
}
总结:
- 源字符串必须以 ‘\0’ 结束。
- 目标空间必须有足够的大,能容纳下源字符串的内容。
- 目标空间必须可修改。
我们有时候需要去比较两个字符串的大小,我们一个一个比较显得格外笨重,那这个时候怎么办呢?
我们就可以使用我们的strcmp函数
strcmp函数是用来比较两个字符串的大小的
int strcmp ( const char * str1, const char * str2 );
1.返回值为int类型
2.两个参数都为const char* 类型
表明两个字符串在这个函数均不会改变仅仅做一个比较
我们的strcmp函数是通过字符的ASCII码值来进行比较的,先比较第一个相同则比较下一个,直到比较出不相同的两个字符,比较ASCII码值谁大谁小
如果两个都遍历到 '\0’的时候依然相等,那么则判断两个字符串相等
如图所示:
接下来我们用代码进行演示:
int main()
{
char arr1[] = "abcde";
char arr2[] = "abcef";
char arr3[] = "abcde";
int a = strcmp(arr1, arr2);
printf("strcmp(abcde,abcef)=%d\n", a);
a = strcmp(arr1, arr3);
printf("strcmp(abcde, abcde)=%d\n", a);
a = strcmp(arr2, arr1);
printf("strcmp(abcef, abcde)=%d", a);
return 0;
}
其实这个函数的使用还是挺简单的。
char *my_strcat(char *dest, const char*src)
{
char *ret = dest;
assert(dest != NULL);
assert(src != NULL);
while(*dest)
{
dest++;
}
while((*dest++ = *src++))
{
;
}
return ret;
}
总结:
- 第一个字符串大于第二个字符串,则返回大于0的数字
- 第一个字符串等于第二个字符串,则返回0
- 第一个字符串小于第二个字符串,则返回小于0的数字
不要弄混了
你有没有觉得这个函数的名字跟另一个函数很相似?
没错,这个函数和我们的strcpy非常相似,而且不仅仅是名字很相似只差一个字,功能也很相似。
strncpy函数的功能就是能实现在源字符串拷贝n个字符到目标空间
这个功能实现了我们想拷贝几个字符都可以,相当于strcpy的升级版,是不是很高级
接下来我们来看看他们之间的用法区别在哪里
char * strncpy ( char * destination, const char * source, size_t num );
1.返回值为char*,返回的是拷贝完毕的字符串
2.参数为目的字符串,源字符串且不可修改,还有一个size_t类型的整型
我们可以猜到大概怎么使用,和前面strcpy差不多,只不过增加了一个整型来限制拷贝
先看代码:
int main()
{
char arr1[20] = "abcd";
char arr2[] = "xxxxxxxx";
strncpy(arr1, arr2, 5);
printf("%s", arr1);
return 0;
}
有了strcpy的基础这个函数使用起来还是很简单的,只不过限制了长度
我们同样也可以将其看做几个部分组成:
合起来就是strncpy,限制长度的字符串拷贝
总结:
- 目标空间必须足够大,以确保能存放拷贝字符串。
- 目标空间必须可变
- 拷贝num个字符从源字符串到目标空间。
- 如果源字符串的长度小于num,则拷贝完源字符串之后,在目标的后边追加0,直到num个。
我们的strcpy可以升级添加一个n限制长度,我们的strcat同样可以升级,升级后变成了strncat
strncat就是从源字符串追加n个字符到目标空间
char * strncat ( char * destination, const char * source, size_t num );
1.返回值为char*,返回追加后的字符串
2.参数为一个目的字符串,源字符串,还有一个无符号整型
看代码:
int main()
{
char a[10] = "aaaa";
char b[] = "bbbb";
strncat(a, b,4);
printf("%s", a);
return 0;
}
我们在使用过程中会发现
我们在输入的无符号整型大于源字符串的长度时,它只会追加到源字符串的最大长度,后面就不会再追加了
总结:
- 目标空间必须有足够的大,能容纳下源字符串的内容。
- 目标空间必须可修改。
不用说,这应该就是取源字符串前n个字符与目标字符串作比较,有了前面几个的经验,很容易就出来了,我们来看一下这个函数的格式以及使用:
int strncmp ( const char * str1, const char * str2, size_t num );
取str2前num个字符与str1作比较
看代码:
int main()
{
char arr1[] = "abcde";
char arr2[] = "abcdff";
int a=strncmp(arr1, arr2, 4);
printf("%d", a);
return 0;
}
总结:
取str2前num个字符与str1作比较
- 大于返回1
- 等于返回0
- 小于返回-1