cstring越界_[转载]C/C++ 内存操作越界略述
不知大家遇到过没有,有时候程序莫名其妙地出现众多问题,但经常调试来调试去,都不知原因所在,我曾经就被这类问题害得很惨,大部分都是内存操作越界引起的。现在就我知道的总结如下,我将其分为两类:一类是无明显表现的,只会给程序运行带来其妙的错误(1、2、3),一类调试时有明显表现(4、5)。
1、字符串数组定义长度忘记加+1(结束符)
如:
char str[10];
::strcpy(str, "0123456789");
此时将会造成越界,因为strcpy将在str为起始地址的第11个单元写入’0’,这样它将覆盖这个单元的值,对于程序来说,有可能带来灾难性的后果。
但,如果你仅仅作字符串操作,一个一个地存入,此时将不会越界,或者你使用strncpy(str, "0123456789",
10),它也不会越界。不过你得小心,此时将不能使用如:strlen,strcpy等等,它的长度和内容变得完全不确定,如果从str地址起的单元都不为‘’,那么将全看作为它的内容,此时对它的操作将会是多么的恐怖!
2、以指针指向数组来进行操作
指针相对数组的下标操作来说,更灵活、更高效,但常常一不小心它就跑到不知地方的地方去了,在循环批量处理中,多移一位就产生了越界,如果是读还好,但如果是写,这就跟上面出现的问题一样了。
3、将对数据类型的错误操作
short int
x;
int y;
x=y
这种操作是不会出错,编译器会解释为 x=(short int)y;丢弃y的高位;但大家看看这个,
void *p
p=&x;
*(int*)p=y;
x只有两个字节的空间,而int需要4个字节的空间,
实际上己出问题了,但你放心,现在的编译器编译后不会引起错误,因为由于内存对齐操作,实际上它分配了4个字节给x,你没有写入到其它数中去,运行会完全正常。
但当你定义的是数组时,那就会有引响了,它会覆盖你后面的数据。
或者你在不同的编程语言中进行数据交换时,而类型与类型宽度不一致就会出问题,如VB中的Integer只有2个字节,而VC中的int是4个,当接口定义不够好时,那就会引发程序崩溃。
4、使用new分配空间,但操作越界
这类越界要容易发现一点,只要你以debug版运行,只要你越界,程序肯定会在delete处停下来。第一次出现此类问题时,真是迷惑不解,我的指针怎么都没变,但就是delete不掉,老报错。由于不知越界会造成此类问题,我一直以为我的电脑哪出毛病了,很正确的事,它就是不对,简直气得我吐血,后来得人指点,才明白,原来这里面还有这个东东!同理malloc,free有此问题。
5、CString在GetBuffer后也容易出现越界
CString确实有很多好用的地方,我最爱它的地方就是作参数,它可以让被调用的函数内分配空间,而在函数外无需对得来的空间进行释放。在CString析构时,它就会帮你释放这块空间。
但在这里面有一个函数你使用不当就会出现越界问题,那就是CString::GetBuffer。通过操作它返回的指针,你便可以直接操作它的缓冲区,这相对通过它的方法来操作内容效率要高出很多,但若你一不小心越界了,那错误也就出来了。不过还好在调试版中,ReleaseBuffer时会报错,若你没有调用ReleaseBuffer,那将在CString析构时报错。
上面4、5之所在调试版时会报错,那是由于它们在释放内存时,都调用delete,而delete的实现如下
void __cdecl operator delete(void* p)
{
#if !defined(_AFX_NO_DEBUG_CRT) &&
defined(_DEBUG)
_free_dbg(p, _NORMAL_BLOCK);
#else
free(p);
#endif
}
_free_dbg(p,
_NORMAL_BLOCK); 将会作很多方面的检查,如果出错,便会报告出来。而free(p);则只是简间的对内存释放
总结
通过上面几点,相信大家知道如何去避免一些内存越界问题,同时也了解内存越界会引发各类问题,在我们遇到一些希奇古怪的问题时,不防关心一下你的内存操作。
cstring越界_[转载]C/C++ 内存操作越界略述相关推荐
- 艾伟_转载:.NET内存管理、垃圾回收
1. Stack和Heap 每个线程对应一个stack,线程创建的时候CLR为其创建这个stack,stack主要作用是记录函数的执行情况.值类型变量(函数的参数.局部变量 等非成员变量)都分配在st ...
- 内存写越界导致破环堆结构引起的崩溃问题定位经验[如报错malloc(): memory corruption或free(): invalid next size]...
前段时间开发的一个后端C模块上线后,线上出core,初始时,因为訪问压力不大,所以崩溃是上线3天左右出现的.当时用gdb跟进调用堆栈并检查源代码,发现出core位置的代码沒有啥问题.因为当时开发任务较 ...
- Marshal 类的内存操作的一般功能
Marshal类 提供了一个方法集,这些方法用于分配非托管内存.复制非托管内存块.将托管类型转换为非托管类型,此外还提供了在与非托管代码交互时使用的其他杂项方法. 命名空间:System.Runtim ...
- 内存溢出_容易造成单片机内存溢出的几个陷阱
[小宅按] 关于程序变量和内存分配,都是需要我们时刻关注的问题.我相信有不少人在这块犯过很多的错误,也可能说明我们基础不够扎实,编写程序的习惯也不够好. 总结一下关于程序的变量和内存方面的概念,虽然是 ...
- C语言处理字符串及内存操作
字符串处理函数 1.字符串长度 strlen表示包含的字符的个数,size_t strlen(char cosnt *string), 返回的是size_t类型,它是无符号整数类型,在表达式中进行运算 ...
- 【转载】ogre内存管理
原文:ogre内存管理 OGRE内存分配策略相关文件及简述 OGRE提供了自己的内存分配策略,甚至为STL容器提供了新的分配策略,相关文件及简述如下: OgreMemoryAllocatedObjec ...
- C#中Marshal 类的内存操作的一般功能概述
Marshal 类的内存操作的一般功能 Marshal类 提供了一个方法集,这些方法用于分配非托管内存.复制非托管内存块.将托管类型转换为非托管类型,此外还提供了在与非托管代码交互时使用的其他杂项方法 ...
- 错误内存【读书笔记】C程序中常见的内存操作有关的典型编程错误
题记:写这篇博客要主是加深自己对错误内存的认识和总结实现算法时的一些验经和训教,如果有错误请指出,万分感谢. 对C/C++程序员来讲,内存管理是个不小的挑战,绝对值得慎之又慎,否则让由上万行代码构成的 ...
- C语言嵌入式系统编程修炼之内存操作
数据指针 在嵌入式系统的编程中,常常要求在特定的内存单元读写内容,汇编有对应的MOV指令,而除C/C++以外的其它编程语言基本没有直接访问绝对地址的能力.在嵌入式系统的实际调试中,多借助C语言指针所具 ...
最新文章
- EIGRP 实验2: 邻居关系
- java丑数算法_LintCode Java算法练习(4)-----丑数II
- python编程入门与案例详解pdf-Python入门之三角函数sin()函数实例详解
- latex自动生成中文目录_自动生成工作表目录三种方法,看哪种适合你?
- 格式引用_论文中如何快速给出规范的文献引用格式?
- Windows批处理命令学习三
- 可视化分析WEB访问:logstalgia
- Firefox必备的24款web开发插件[转]
- 操作系统第二章总结/
- 交换二叉树的每个节点的左右子树
- 什么是SQL Server数据库镜像?
- 程序员界之行业求职黑名单!实用!
- 新手用canvas画时钟
- py----Geany编辑器的安装_配置与使用
- LTE中的RSRP、RSSI、RSRQ、SINR、MCS介绍
- python实现账号密码登录
- Envato不停机迁移边缘网络提供商
- Live555: RtspServer 示例
- markdown之mermaid
- 【转载】 C#实现的CRC32算法
热门文章
- Git 分支管理-创建与合并
- unity编辑器扩展#3 《Extending Unity with Editor Scripting 》笔记
- SwiftUI 2020年开源项目和教程合集
- 10年前被删的初恋,凌晨1点突然加我…
- java中字符流和字节流的区别_java中字节流和字符流有哪些区别
- vue.js实现图片、视频文件压缩后再上传
- 【180629】VC++滚彩球游戏附源程序
- ApacheCON Asia 2021清华大学软件学院王建民:工业大数据软件与开源创新
- 免费申请极验GEETEST4.0行为验证账号
- 美国随着疫情创新高的,还有十亿身价富豪的人数……