【 C 】高级字符串查找之 strspn 和 strcspn 的思考

举报
李锐博恩 发表于 2021/07/15 08:18:50 2021/07/15
【摘要】 strspn 这个库函数是真的很难理解,看了很多中文描述,反正都是不知所云,给出一系列的例子,结果也是让我瞠目结舌,荒唐、荒谬、荒诞! 特此记录理解过程,最后竟然是百度百科让我明白了这个库函数的意思。 中文的描述真的是太困难了,想用一两句话去说清楚往往不知所云。百度百科上最后给出了这个函数的定义,也就是这个函数是如何实现的,看了几遍才恍然大悟! 因此我也按照这个理解的顺...

strspn 这个库函数是真的很难理解,看了很多中文描述,反正都是不知所云,给出一系列的例子,结果也是让我瞠目结舌,荒唐、荒谬、荒诞!

特此记录理解过程,最后竟然是百度百科让我明白了这个库函数的意思。

中文的描述真的是太困难了,想用一两句话去说清楚往往不知所云。百度百科上最后给出了这个函数的定义,也就是这个函数是如何实现的,看了几遍才恍然大悟!

因此我也按照这个理解的顺序给出解释(没有耐心读的,不要往下看了,用心的东西往往需要花时间体会!),先给出函数的定义:


  
  1. int strspn(const char *s,const char *accept)
  2. {
  3. const char *p;
  4. const char *a;
  5. int count = 0;
  6. for(p = s; *p != '\0'; ++p)
  7. {
  8. for (a = accept; *a != '\0'; ++a)
  9. {
  10. if (*p == *a)
  11. {
  12. ++count;
  13. break;
  14. }
  15. } //里面的for循环到此为止
  16. if (*a == '\0')
  17. {
  18. return count;
  19. }
  20. } //外面的for循环到此为止
  21. return count;
  22. }

可以这么说,如果你看懂了这个函数定义,难道还不能理解这个函数?

比任何的中文解释更清晰!

那关键是你能看懂这个函数,如果看不懂那就回去再补补基础。

声明:下面这段中文解释如果看不懂,那就自己理解上面的程序,别看了,有些东西只可意会不可言传!

这里为了方便看,对上面函数的定义的要点提出来几点:

关键的是for循环语句的嵌套,内部for函数:


   
  1. for (a = accept; *a != '\0'; ++a)
  2. {
  3. if (*p == *a)
  4. {
  5. ++count;
  6. break;
  7. }
  8. } //里面的for循环到此为止

什么意思呢?就是将accept这个字符数组中的所有字符与目的字符数组 s 中的字符比较,如果相等,则计数器加1,那比较到什么时候结束呢?

比较到accept中的所有字符没有一个与s中字符相等,这时执行内层for循环后面的一条语句:


   
  1. if (*a == '\0')
  2. {
  3. return count;
  4. }

这不就是直接返回一个值,就是至此为止的计数值。


当你使用这个函数的使用,当然不需要自己定义,因为这是库函数中定义的,所以,你只需要源程序开头带上#include<string.h>这个头文件即可!

下面举几个例子,看看测试结果:

这是第一个测试的例子:


  
  1. #include <stdio.h>
  2. #include <string.h>
  3. int main () {
  4. int len;
  5. const char str1[] = "25,142,330,Smith,J,239-4123";
  6. const char str2[] = ",0123456789";
  7. len = strspn(str1, str2);
  8. printf("Length of initial segment matching %d\n", len );
  9. return(0);
  10. }

结果为:

在这个地方,发现了一个神器,在线写程序并编译的地址:Online C Compiler

http://tpcg.io/luVeGa

再给出一个测试例子:


  
  1. #include <stdio.h>
  2. #include <string.h>
  3. int main () {
  4. int len;
  5. const char str1[] = "25,142,330,Smith,J,239-4123";
  6. const char str2[] = "0123456789";
  7. len = strspn(str1, str2);
  8. printf("Length of initial segment matching %d\n", len );
  9. return(0);
  10. }


对了,还有一个于此互补的函数 strcspn,它的作用与上面的函数互补,为了说明清楚,我们给出两个函数的函数原型,便于讨论:

size_t strspn( char const *str, char const *group );

size_t strcspn( char const *str, char const *group );

group字符串指定一个或多个字符。strspn返回str起始部分匹配 group 中任意字符的字符数。这只是一个不太清晰的总结,具体看上面的内容。

strcspn函数和strspn函数正好相反,它对str字符串起始部分中不与group 中任何字符匹配的字符进行计数。 strcspn 这个名字中字母c来源于一组字符求补这个概念,也就是把这些字符换成原先并不存在的字符。

上面这一小段来自于《C与指针》,仅供参考,我也不太明白它在说什么?

举几个例子说明一下吧:


  
  1. #include <stdio.h>
  2. #include <string.h>
  3. int main () {
  4. int len;
  5. const char str1[] = ",142,330,Smith,J,239-4123";
  6. const char str2[] = "0123456789";
  7. len = strcspn(str1, str2);
  8. printf("Length of initial segment matching %d\n", len );
  9. return(0);
  10. }

str2字符数组中每一个字符都不与str1字符数组中的第一个字符相等,故计数加1,然后str2中有字符与str1第2个字符匹配的了,说了函数直接返回结果为1.

为了与之对比,我在举一个例子:


  
  1. #include <stdio.h>
  2. #include <string.h>
  3. int main () {
  4. int len;
  5. const char str1[] = ",,,Smith,J,239-4123";
  6. const char str2[] = "0123456789";
  7. len = strcspn(str1, str2);
  8. printf("Length of initial segment matching %d\n", len );
  9. return(0);
  10. }


这里根据我的想法,我想给出strcspn的大致定义,应该是这样的:


  
  1. int strcspn(const char *s,const char *accept)
  2. {
  3. const char *p;
  4. const char *a;
  5. int count = 0;
  6. for(p = s; *p != '\0'; ++p)
  7. {
  8. for (a = accept; *a != '\0'; ++a)
  9. {
  10. if (*p == *a)
  11. {
  12. return count;
  13. }
  14. } //里面的for循环到此为止
  15. if (*a == '\0')
  16. {
  17. ++ count;
  18. }
  19. } //外面的for循环到此为止
  20. return count;
  21. }

不信的话,下面我来测试下它的功能:


  
  1. #include <stdio.h>
  2. #include <string.h>
  3. int main () {
  4. int len;
  5. const char str1[] = ",,,Smith,J,239-4123";
  6. const char str2[] = "0123456789";
  7. //函数原型
  8. int my_strcspn(const char *s,const char *accept);
  9. len = my_strcspn(str1, str2);
  10. printf("Length of initial segment matching %d\n", len );
  11. return(0);
  12. }
  13. int my_strcspn(const char *s,const char *accept)
  14. {
  15. const char *p;
  16. const char *a;
  17. int count = 0;
  18. for(p = s; *p != '\0'; ++p)
  19. {
  20. for (a = accept; *a != '\0'; ++a)
  21. {
  22. if (*p == *a)
  23. {
  24. return count;
  25. }
  26. } //里面的for循环到此为止
  27. if (*a == '\0')
  28. {
  29. ++ count;
  30. }
  31. } //外面的for循环到此为止
  32. return count;
  33. }

运行得到:

运行示意

可见,结果符合预期!

就到这里吧,最后给出几个不错的网址:

https://baike.baidu.com/item/strspn

https://www.tutorialspoint.com/c_standard_library/c_function_strspn.htm

 

文章来源: reborn.blog.csdn.net,作者:李锐博恩,版权归原作者所有,如需转载,请联系作者。

原文链接:reborn.blog.csdn.net/article/details/82562195

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。