C语言strlen()以NUL作为字符串结束标记,自定义一个字符串长度统计函数消除这个Bug

分享到:
QQ空间
新浪微博
腾讯微博
豆瓣
人人网
我们知道,字符串长度统计函数 strlen() 以NUL作为字符串结束标记,但是很不幸的是,有时候字符串并不以NUL结束,例如:

复制纯文本复制

  1. char strA[5] = {'1', '2', '3', '4', '5'};
char strA[5] = {'1', '2', '3', '4', '5'};

这个时候使用 strlen() 函数会出现莫名其妙的结果,因为 strlen() 会一直统计内存中的字符,直到遇到NUL,而什么时候遇到是不可预知的。

请大家编写一个安全的字符串长度统计函数,即使字符串未以NUL结尾,也不会出错。

注意:你需要向自定义函数传递一个参数,它的值就是字符串缓冲区的长度。

复制纯文本复制

  1. #include <stdio.h>
  2. #include <string.h>
  3. #include <stddef.h>
  4. /**
  5. * @function 安全的字符串长度函数
  6. *
  7. * @param string 要计算长度的字符串
  8. * @param size 字符串缓冲区长度
  9. **/
  10. size_t my_strlen( char const *string, size_t size ){
  11. register size_t length;
  12. // 统计范围不超过 size
  13. for ( length = 0; length < size; length++ )
  14. if( *string++ == '\0')
  15. break;
  16. return length;
  17. }
  18. int main(){
  19. char strA[5] = {'1', '2', '3', '4', '5'},
  20. strB[10] = "123456789";
  21. // 对比 strlen() 和 my_strlen() 的结果
  22. printf("The length of strA is %d -- By strlen()\n", strlen(strA) );
  23. printf("The length of strA is %d -- By my_strlen()\n", my_strlen(strA, 5) );
  24. printf("The length of strB is %d -- By strlen()\n", strlen(strB) );
  25. printf("The length of strB is %d -- By my_strlen()\n", my_strlen(strB, 10) );
  26. return 0;
  27. }
#include <stdio.h>
#include <string.h>
#include <stddef.h>/*** @function 安全的字符串长度函数* * @param  string  要计算长度的字符串* @param  size  字符串缓冲区长度
**/
size_t my_strlen( char const *string, size_t size ){register size_t length;// 统计范围不超过 sizefor ( length = 0; length < size; length++ )if( *string++ == '\0')break;return length;
}int main(){char strA[5] = {'1', '2', '3', '4', '5'},strB[10] = "123456789";// 对比 strlen() 和 my_strlen() 的结果printf("The length of strA is %d -- By strlen()\n", strlen(strA) );printf("The length of strA is %d -- By my_strlen()\n", my_strlen(strA, 5) );printf("The length of strB is %d -- By strlen()\n", strlen(strB) );printf("The length of strB is %d -- By my_strlen()\n", my_strlen(strB, 10) );return 0;
}

运行结果:

The length of strA is 17 -- By strlen()
The length of strA is 5 -- By my_strlen()
The length of strB is 9 -- By strlen()
The length of strB is 9 -- By my_strlen()

注意:17 明显超出了字符数组范围。

C语言实现动态数组,克服静态数组大小固定的缺陷

分享到:
QQ空间
新浪微博
腾讯微博
豆瓣
人人网
C语言中,数组长度必须在创建数组时指定,并且只能是一个常数,不能是变量。一旦定义了一个数组,系统将为它分配一个固定大小的空间,以后不能改变,称为静态数组。但在编程过程中,有时我们所需的内存空间无法预先确定,对于这个问题,用静态数组的办法很难解决。

动态数组是相对于静态数组而言。静态数组的长度是预先定义好的,在整个程序中,一旦给定大小后就无法改变。而动态数组则不然,它可以随程序需要而重新指定大小。动态数组的内存空间是从堆(heap)上分配(即动态分配)的。是通过执行代码而为其分配存储空间。当程序执行到这些语句时,才为其分配。程序员自己负责释放内存。

那么,如何创建动态数组,按照需要设置数组大小呢?

下面是一个创建动态数组的例子:

复制纯文本复制

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. int main(){
  4. int arrLen; // 数组长度
  5. int *array; // 数组指针
  6. int i; // 数组下标
  7. printf("输入数组长度:");
  8. scanf("%d", &arrLen);
  9. // 动态分配内存空间,如果失败就退出程序
  10. array = (int*)malloc( arrLen*sizeof(int) );
  11. if(!array){
  12. printf("创建数组失败!\n");
  13. exit(1);
  14. }
  15. // 向内存中写入数据
  16. for(i=0; i<arrLen; i++){
  17. array[i] = i+1;
  18. }
  19. // 循环输出数组元素
  20. for(i=0; i<arrLen; i++){
  21. printf("%d ", array[i]);
  22. }
  23. printf("\n");
  24. free(array);
  25. system("pause");
  26. return 0;
  27. }
#include <stdio.h>
#include <stdlib.h>int main(){int arrLen;  // 数组长度int *array;  // 数组指针int i;  // 数组下标printf("输入数组长度:");scanf("%d", &arrLen);// 动态分配内存空间,如果失败就退出程序array = (int*)malloc( arrLen*sizeof(int) );if(!array){printf("创建数组失败!\n");exit(1); }// 向内存中写入数据for(i=0; i<arrLen; i++){array[i] = i+1;}// 循环输出数组元素for(i=0; i<arrLen; i++){printf("%d  ", array[i]);}printf("\n");free(array); system("pause");return 0;
}

运行结果:
输入数组长度:10
1 2 3 4 5 6 7 8 9 10
请按任意键继续. . .

malloc()函数

这里重点说明的是malloc()函数,这是一个非常重要和常用的函数。

malloc() 用来动态分配指定大小的内存空间,以字节计,其原型为:
   void *malloc( size_t size );
size_t 是一种自定义数据类型,在 stddef.h 头文件中定义为:
   typedef unsigned int size_t;  // 无符号整型

malloc()返回值类型为 void *,这并不是说该函数调用后无返回值,而是返回一个内存结点的地址,该地址的类型为void(无类型或类型不确定),即一段存储区的首址,其具体类型无法确定,只有使用时根据各个域值数据再确定。可以用强制转换的方法将其转换为别的类型。例如:

复制纯文本复制

  1. double*pd=NULL;
  2. pd=(double*)malloc(10*sizeof(double));
double*pd=NULL;
pd=(double*)malloc(10*sizeof(double));

表示将向系统申请10个连续的double类型的存储空间,并用指针pd指向这个连续的空间的首地址。并且用(double)对malloc()的返回类型进行转换,以便把double类型数据的地址赋值给指针pd。

sizeof

上面的代码中,array = (int*)malloc( arrLen*sizeof(int) ); 用来分配arrLen*sizeof(int)个字节的内存空间,并将返回的指针强制转换为int。这里注意,int 类型的长度在不同平台下可能不同,不要把int的长度指定为2或4,要用sizeof()来计算int的长度,以更好的跨平台。

使用内存中的数据

上面的代码中,我们通过下标(array[i])来引用数组元素,这个静态数组没有什么区别。另外还可以通过指针来引用数组元素,对上面的程序稍作修改:

复制纯文本复制

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. int main(){
  4. int arrLen; // 数组长度
  5. int *array; // 数组指针
  6. int *arrayCopy; // 数组指针副本
  7. int i; // 数组下标
  8. printf("输入数组长度:");
  9. scanf("%d", &arrLen);
  10. // 动态分配内存空间,如果失败就退出程序
  11. arrayCopy = array = (int*)malloc( arrLen*sizeof(int) );
  12. if(!array){
  13. printf("创建数组失败!\n");
  14. exit(1);
  15. }
  16. // 向内存中写入数据
  17. for(i=0; i<arrLen; i++){
  18. *arrayCopy++ = i+1;
  19. }
  20. // 循环输出数组元素
  21. arrayCopy = array;
  22. for(i=0; i<arrLen; i++){
  23. printf("%d ", *arrayCopy++);
  24. }
  25. printf("\n");
  26. free(array);
  27. system("pause");
  28. return 0;
  29. }
#include <stdio.h>
#include <stdlib.h>int main(){int arrLen;  // 数组长度int *array;  // 数组指针int *arrayCopy;  // 数组指针副本 int i;  // 数组下标printf("输入数组长度:");scanf("%d", &arrLen);// 动态分配内存空间,如果失败就退出程序arrayCopy = array = (int*)malloc( arrLen*sizeof(int) );if(!array){printf("创建数组失败!\n");exit(1); }// 向内存中写入数据 for(i=0; i<arrLen; i++){*arrayCopy++ = i+1;}// 循环输出数组元素arrayCopy = array;for(i=0; i<arrLen; i++){printf("%d  ", *arrayCopy++);}printf("\n");free(array); system("pause");return 0;
}

可以发现,我们必须另外定义一个指针变量 arrayCopy,用来指向具体的数组元素。数组赋值完成后,要将 arrayCopy 重置到数组首地址,以便后面循环输出。

这里注意:free() 函数必须释放整块内存,不能只释放一部分,或者释放不存在的内存空间,否则程序会出错。所以,要多定义一个变量 arrayCopy,不断改变它的值,以指向不同的数组元素。这样可以保证 array 变量的值不变,始终指向内存首地址,用于free()整块内存。

例如,如果改变了 array 的值,使其指向第5个数组元素,那在free(array)时只释放掉了后半部分内存,而没有释放掉前半部分内存,这将引起程序错误。

strlen() Bug相关推荐

  1. C语言再学习 -- 字符串和字符串函数

    最近身体很不给力,很乏累!白天没精神,晚上睡不着,心情还很烦躁.看书都有点看不下去的样子,到了C语言最难掌握的部分了,数组.指针.字符串.硬着头皮看书总结吧. 一.字符串 1.字符串介绍 字符串是以空 ...

  2. C++接收字符串数组_C语言处理字符串的7个函数

    C库提供了多个处理字符串的函数,ANSI-C把这些函数的原型放在string.h头文件中.其中最常用的函数有strlen().strcat().strcmp().strncmp().strcpy()和 ...

  3. C Primer Plus 第11章_字符串和字符串函数_代码和练习题

    11.1 表示字符串和字符串I/O 字符串是以空字符(\0)结尾的char类型数据. strings1.c 演示在程序中表示字符串的几种方式 #include <stdio.h> #def ...

  4. C primer plus(第六版)第十一章源代码

    C primer plus(第六版)第十一章源代码 /* 11.1 */ #include<stdio.h> #define MSG "I am a symbolic strin ...

  5. C语言基础之11:字符串和字符串函数

    Tips1: 函数:gets().gets_s().fgets().puts().fputs().strcat().strncat(). strcmp().strncmp().strcpy().str ...

  6. bug诞生记——临时变量、栈变量导致的双杀

    这是<bug诞生记>的第一篇文章.本来想起个文艺点的名字,比如<Satan(撒旦)来了>,但是最后还是想让这系列的重心放在"bug的产生过程"和" ...

  7. 比较分析与数组相关的sizeof和strlen

    // 形如: int a[]={1,2,3,4,5}; char name[]="abcdef"; 无论是整型数组还是字符数组,数组名作为右值的时候都代表数组首元素的首地址. 数组 ...

  8. 人人都写过的5个Bug!

    计算机专业的小伙伴,在学校期间一定学过 C 语言.它是众多高级语言的鼻祖,深入学习这门语言会对计算机原理.操作系统.内存管理等等底层相关的知识会有更深入的了解,所以我在直播的时候,多次强调大家一定要好 ...

  9. opensips中db_default_url解析存在的bug

    错误现象 22:29:55 [86] DBG:db_mysql:db_mysql_connect: opening connection: mysql://xxxx:xxxx@:gsf@1202]@1 ...

最新文章

  1. 马斯克的90后合伙人离职Neuralink!5岁学编程,曾创立2家公司
  2. 【Tools】Modbus Slave 7安装详解
  3. PowerBI,自定义编辑同一页面中不同图表之间的交互,使页面交互更灵活
  4. SQL Server经典查询语句练习题及答案
  5. android.mk添加静态库,Android Gradle添加静态库
  6. Codeforces Round #381 (Div. 2)
  7. Vivado时序报告名词解释
  8. Windows Terminal v0.7 发布:支持分屏、重排选项卡和改进 UI
  9. SAP License:SAP合同类型的使用
  10. 虚幻学习4---制作实时逼真的毛发【笔记】
  11. pygame安装教程(window)
  12. HTML加载图片跨域
  13. 你知道怎么下载矢量图标吗——Iconfont
  14. java游戏繁体字名字,繁体字游戏名字你知道有几个?
  15. Ubuntu发烧友三部曲
  16. electron-builder 打包 exe 异常错误集锦
  17. K8s——kubernetes集群中ceph集群使用【下】
  18. 第一部分 利用opencv基于hsv颜色的目标提取效果和python代码
  19. 以mysql为例有几种隔离级别_mysql有几种隔离级别
  20. 电源并联均流--UC3907,UC3902

热门文章

  1. cmd python封装成exe_别再问我怎么Python打包成exe了!
  2. linux 日志切割 自带,[日志分割回滚] 使用linux自带的logrotate对nginx日志进行分割...
  3. mysql 事物 锁行 测试_MySQL Transaction--RR事务隔离级别下加锁测试
  4. python进程监控 supervisor_使用Python的Supervisor进行进程监控以及自动启动
  5. sequelize 外键关联_mysql – Sequelize.js外键
  6. 视图的数据存放在哪里_分布式 | DBLE 是如何实现视图的?
  7. 从windows计算机中卸载office,如何彻底卸载电脑中的Office组件|Office办公软件卸载不干净怎么办...
  8. python画正方形内切圆_python画出三角形外接圆和内切圆的方法
  9. java接口文档生成工具_接口文档生成
  10. python做var模型_【Python金融量化】VaR系列(五):Copula模型估计组合VaR-阿里云开发者社区...