2017/03/11学习笔记

.lib资源模式文件 描述 .dll
.dll 动态库(函数二进制码集合,里面有函数的函数体),动态库有规范(win/linux)下不一样.

mtrace

是linux C编程内存泄漏检测工具,mtrace是三款工具中最简单易用的,mtrace是一个C函数,在<mckeck.h>里面声明及定义.
其实mtrace是类似malloc_hook的malloc handler,只不过mtrace的hander function已由系统为你写好,既然如此,系统又怎么知道您想将malloc/free的记录写在哪里?为此,调用mtrace前要设置MALLOC_TRACE环境变量.

setenv("MALLOC_TRACE","output_file_name",1);

output_file_name就是存储检测结果的文件名称。

但是检测结果的格式一般人无法理解,而只要安装mtrace的话,就会有一名为mtrace的Perl script,在sheel下输入:
mtrace [binary] output_file_name

Memwatch

Mem设置最简单.他能检测未释放的内存、同一段内存被多次释放、位地址存取错误及不当使用为分配的内存。
MemWatch将所有的分配内存用0xFE填充,所以,如果你看到错误的数据是0xFE,那就是你没有初始化数据。
MemWatch将所有已释放的内存用0xFD填充,如果你发现你使用的数据是0xFD,那就是使用已释放的内存。在这种情况下,MemWatch会立即把一个释放了的块信息填在释放了的数据前。这个块包括关于内存在哪儿释放的信息,以可读文本形式存放。

Win32下动态链接库(DLL)编程原理

比较大的应用程序都有很多模块组成,这些模块分别完成相对独立的功能,他们彼此协作来完成整个软件系统的工作。其中可能存在一些模块的功能比较通用,在构造其他软件系统时仍会被使用,在构造软件系统时,如果将所有的模块的源代码都静态额编译到整个程序中,会产生一些问题:一个是增加了程序的大小,要占用更多的磁盘空间,程序运行的时候也会消耗较大的内存。另一个问题是,在编写大的程序是,每次修改重建都必须编译所有源代码,增加编译的复杂性,也不利于阶段性的单元测试。

动态库内存释放问题

1.保证内存分配和释放的统一性:如果一个DLL提供一个能够分配内存的函数,那么这个DLL同事应该提供一个函数释放这些内存。数据的创建和释放应该在同一个层次上。
曾经遇到这样的例子:在dll中分配了内存,通过PostMEssage将其地址传给应用。然后去释放他,总是报异常。
2.如果EXE用MFC Appwizard方式生成,dll用wind32方式生成,则运行时会查询错误。进一步用单步跟踪,发现mfc方式和win32下的new操作符是用不同方式实现的,源程序分别在VC目录文件Afmem.cpp和new.cpp中。
因为DLL输出函数后,并不知道是哪一模块调用他,因此new和delete配对是最好在一个文件中。
3.问题主要在于DLL和EXE主程序分配内存的堆不一样,你可以不用 new和delete,而是使用
1)::HeapAlloc(::GetProcessHeap(),...)和::HeapFree(::GetProcessHeap(),...)
2)::GlobalAlloc()和GlobalFree()
这两点API,这样无论在DLL中还是主程序中都是在进程默认堆中分配,就不会出错了(JNI容易出错SSH项目中出错,不稳定)。
4.还有一个办法,就是把DLL的Settings的c/c++选项卡的Code Generation的Use Run-time liberary改成Debug Multitheaded DLL,在Release版本改成Multithread DLL,就可以直接使用new 和delete了。不过MFC就不能使用Shared模式了。

时间: 2024-10-05 04:33:53

2017/03/11学习笔记的相关文章

2017/03/31学习笔记

双向链表 单向链表的节点都只有一个指向下一个节点的指针单向链表的数据元素无法直接访问其前驱元素逆序访问单向链表中的元素时极其耗时的操作双向链表在单向链表的基础上增加了指向前驱的指针功能上双向链表可以完全取代单向链表的使用 栈是一种特殊的线性表 栈仅能在线性表的一端进行操作栈顶:允许操作的一端栈底:不允许操作的一端首先它是一个线性表,也就是说,栈元素具有线性关系,即前驱后继关系.只不过它是一种特殊的线性表.定义中说是在线性表的表尾进行插入和删除操作,这里表尾是指栈顶,而不是栈底.他的特殊之处就在于

2017/03/27学习笔记

程序的输入是指从输入文件讲数据传送给程序,程序的输出是指从程序将数据传送输出文件.C++输入输出包含以下三方面内容:对系统指定标准设备的输入和输出.即从键盘输入数据,输出到显示器.这种输入输出称为标准输入输出,简称标准IO.以外出磁盘文件为对象进行输入输出,即从磁盘文件输入数据,将数据输出到文件.以外存为对象的输入输出称为文件的输入输出,简称文件IO.度内存中指定的空间进行输入输出,通常指定一个字符串数组作为储存空间(实际上可以利用该空间储存任何信息).这种输入输出称为字符串输入输出,简称串IO

2017/03/24学习笔记

类型转换 C语言中的强制类型转换很简单,不管什么类型的转换都是TYPE b=(TYPE)a; C++中类型转换提供了4种类型转换操作符来应对不同场合的应用.static_cast 静态转换.如int转换成charreinterpreter_cast 重新解释类型dynamic_cast 命名上理解是动态类型转换.如子类和父类之间的多态类型转换 (父类转子类,向下转型)const_cast 字面理解就是去const属性. 4种类型转换格式TYPE b=static_cast<TYPE>(a);

2017/03/17学习笔记

智能指针类 指针使用过程中,经常会出现内存泄漏和内存多次被释放.解决方案:boost库的智能指针项目开发中,要求开发者使用预先编写的智能指针类对象代替C语言中原生的指针.智能指针思想工程中的智能指针是一个类模板通过构造函数接管申请的内存通过析构函数确保堆内存被及时释放通过重载指针运算符*和->来模拟指针的行为通过重载比较运算符==和!=来模拟指针的比较 为什么不要重载&&和||操作符 &&和||是C++中非常特殊的操作符&&和||内置实现了短路规则操作

2017/03/19学习笔记

继承中的同名成员变量处理方法 1.当子类成员变量和父类成员变量同名时2.子类依然从父类继承同名变量3.在子类中通过作用域分辨符::进行同名成员区分(在子类中使用父类的同名成员,显式地使用类名限定符)4.同名成员存储在内存中的不同位置 继承中的static关键字 继承和stataic关键字在一起会产生什么现象?1.父类定义的静态成员,将被子类所共享2.根据静态成员自身的访问特性和子类的继承方式,在类层次体系中具有不同的访问性质(遵守子类的访问控制)3.子类中访问静态成员,用以下形式:类名::成员或

2017/03/13学习笔记

cout<<标准输出cin>>标准输入定义一个类,是一个抽象概念,不会给你分配内存.用数据类型定义变量的时候,才会分配内存. C语言是在实践的过程中逐步完善起来的.没有深思熟虑的设计过程.使用时存在很多的"灰色地带"残留量过多低级语言的特征直接利用指针进行内存操作.C语言的目标是高效最终程序执行效率的高效C语言和C++并不是对立的竞争关系.C++ 是C语言的加强,是一种更好的C语言.C++ 是以C语言的为基础,并且完全兼容C语言的特性 C语言中无法取得regis

2017/03/09学习笔记

加密分为:对称加密和非对称加密对称加密:加密的秘钥和解密的密钥一样运算速度快,适合做大数据加密,强度弱eg :des 3des AES196 512....非对称加密:加密的密钥和解密的密钥不一样.运算速度慢适合做小数据加密,强度大 安全eg:rsa1024 2048 ECC椭圆曲线(SM系列)加密三要素:明文 密文 密钥 散列 哈希(指纹)(不可逆) 对称加密的原理:分组(按密钥长度分组,最后以加密 padding方式打补丁)

2017/03/08学习笔记

结构体深copy和浅copy问题 编译器的=号操作,只会把结构体中指针变量的值copy,但不会重新开辟内存空间.//出现浅copy出现的场景是,结构体里面有指针成员的时候.如果需要执行深copy,需要手动显示分配内存,然后手动copy指针成员所执行的数据. 结构体偏移量定义下来,则结构体中的成员 内存布局就定义下来了,可以通过成员地址去求结构体的内存地址 文件操作 文件的分类按文件的逻辑结构分为记录文件:由具有一定结构的记录组成(定长和不定长)流式文件:由一个个字符(字节)数据顺序组成.按存储介

2017/03/16学习笔记

//void oper(int a,int b) const ;==>void oper(const className* const this,int a,int b); 运算符重载 所谓重载,就是重新赋予新的含义.函数重载就是对一个已有的函数赋予新的含义,使之实现新功能,因此,一个函数名就可以用来代表不同的功能函数,也就是一名多用.运算符也可以重载.实际上,我们已经不知不觉中使用了运算符重载.如,大家都习惯用加法运算符"+"对整数.浮点数进行加法运算 5+8.5.8+3.67