技术回归01-Windows内存分配工具
很久没有写技术方面的东西了,这半年主要是在学习别人的东西,对自己提高比较大,算是一次技术回笼吧,这次学习之旅目的是结束技术方面的专注,开始向应用方面找突破口,也就是完成技术积累或者为技术的积累做坚实的准备。
c/C++的一个让人疯狂的地方就是内存管理,非法访问、越界、野指针、泄漏、内存分配器等诸多问题,有时候一个编程老手也会迷惘困惑。Crt有一些堆栈检查的函数可以完成基本的内存状况检查,MFC也有一些简单的对象检查机制,当然好的算是java、.net等sdk的超重量级封装了,即使发生对象错误也能把堆栈信息明明白白的告诉你(至少表面上是这样,具体我对这两种语言没有做过开发)。下面介绍的是某牛公司实现的内存分配工具,基本实现了内存泄漏检查,对象合法性检查,对于我来说已经够用了。
为了对内存分配块进行跟踪,设计如下结构体:
//+--------------------------------------------------------------
//
// 每个请求分配内存块的前缀结构体
// 用来跟踪所有请求分配块以及请求分配名称
//
//---------------------------------------------------------------
struct DBGALLOCHDR
{
DBGALLOCHDR* pdbgahPrev; // 前一个内存块头
DBGALLOCHDR* pdbgahNext; // 后一个内存块头
DWORD iAllocated; // 记录是第几次请求分配操作
DWORD tid; // 请求分配线程的ID
size_t cbRequest; // 请求分配大小
char szName[64]; // 请求分配块名称
DWORD adwGuard[4];// 保护头
};
//+--------------------------------------------------------------
//
// 每个请求分配内存块的后缀结构体
// 使用特定的数据填充用来检测指针是合法
//
//---------------------------------------------------------------
struct DBGALLOCFOOT
{
DWORD adwGuard[4];
};
// 内存跟踪块的根,通过根可以获取所有分配块
DBGALLOCHDR g_dbgahRoot =
{
&g_dbgahRoot,
&g_dbgahRoot,
0,
(DWORD)-1
};为了实现多线程内存分配跟踪,采用Tls技术使用线程局部对象保存当前分配信息:
// 线程局部对象结构体,辅助实现每个线程的请求内存分配记录
struct DBGTHREADSTATE
{
DBGTHREADSTATE* ptsNext;
DBGTHREADSTATE* ptsPrev;
// Add globals below
void* pvRequest; // 线程最后一次请求分配内存的指针
size_t cbRequest; // 线程最后一次请求分配内存的大小
};


// 调试期间实际分配内存大小=请求分配+分配头+分配尾
size_t _ActualSizeFromRequestSize(size_t cb)
{
return cb+sizeof(DBGALLOCHDR)+sizeof(DBGALLOCFOOT);
}主要实现的内存分配工具有如下这些:
void* _MemAlloc(ULONG cb);
void* _MemAllocClear(ULONG cb);
HRESULT _MemRealloc(void** ppv, ULONG cb);
ULONG _MemGetSize(void* pv);
void _MemFree(void* pv);
HRESULT _MemAllocString(LPCTSTR pchSrc, LPTSTR* ppchDst);
HRESULT _MemAllocString(ULONG cch, LPCTSTR pchSrc, LPTSTR* ppchDst);
HRESULT _MemReplaceString(LPCTSTR pchSrc, LPTSTR* ppchDest);
#define MemAlloc(cb) _MemAlloc(cb)
#define MemAllocClear(cb) _MemAllocClear(cb)
#define MemRealloc(ppv, cb) _MemRealloc(ppv, cb)
#define MemGetSize(pv) _MemGetSize(pv)
#define MemFree(pv) _MemFree(pv)
#define MemAllocString(pch, ppch) _MemAllocString(pch, ppch)
#define MemAllocStringBuffer(cch, pch, ppch) _MemAllocString(cch, pch, ppch)
#define MemReplaceString(pch, ppch) _MemReplaceString(pch, ppch)
#define MemFreeString(pch) _MemFree(pch)通过宏实现类的new delete重写:
#define DECLARE_MEMALLOC_NEW_DELETE() \
inline void* __cdecl operator new(size_t cb) { return(MemAlloc(cb)); } \
inline void* __cdecl operator new[](size_t cb) { return(MemAlloc(cb)); } \
inline void __cdecl operator delete(void* pv) { MemFree(pv); }
#define DECLARE_MEMCLEAR_NEW_DELETE() \
inline void* __cdecl operator new(size_t cb) { return(MemAllocClear(cb)); } \
inline void* __cdecl operator new[](size_t cb) { return(MemAllocClear(cb)); } \
inline void __cdecl operator delete(void* pv) { MemFree(pv); }在应用的时候可以重写全局new delete:
// 测试全局new delete
void* __cdecl operator new(size_t cb) { return(MemAlloc(cb)); }
void* __cdecl operator new[](size_t cb) { return(MemAlloc(cb)); }
void __cdecl operator delete(void* pv) { MemFree(pv); }使用注意:
进程启动时候需要调用:
_DbgDllProcessAttach();
_afxGlobalData._hProcessHeap = GetProcessHeap();
进程退出的时候需要调用:
_DbgDllProcessDetach();
测试用例:
// 测试基本类型
void TestBuiltin()
{
// 基本类型
int* pInt = new int(10);
int* pIntAry = new int[10];
char* pStr = new char[100];
MemSetName((pStr, "String"));
}
// 测试class
void TestClass()
{
Cls* pCls = new Cls();
}
// 测试释放
void TestOk()
{
Cls* pCls = new Cls();
delete pCls;
pCls = NULL;
}
DWORD WINAPI ThreadProc(LPVOID lpParameter)
{
int* pIntAry = new int[100];
return 0;
}
// 测试多线程
void TestMultiThread()
{
HANDLE hHandle = CreateThread(NULL, 0, ThreadProc, NULL, 0, NULL);
WaitForSingleObject(hHandle, -1);
}
int main(int argc, char* argv[])
{
_DbgDllProcessAttach();
_afxGlobalData._hProcessHeap = GetProcessHeap();
TestBuiltin();
TestClass();
TestMultiThread();
TestOk();
_DbgDllProcessDetach();
return 0;
}调试输出窗口结果:
A + 4 - 0 = [ 4]
A + 40 - 0 = [ 44]
A + 100 - 0 = [ 144]
A + 8 - 0 = [ 152]
A + 400 - 0 = [ 552]
The thread 0x1D38 has exited with code 0 (0x0).
A + 8 - 0 = [ 560]
F + 0 - 8 = [ 552]
---------- Leaked Memory Blocks ----------
p=0x00144354 cb=400 #=4 TID:0x1d38
p=0x00144294 cb=8 #=3 TID:0x1878
p=0x001441a4 cb=100 #=2 TID:0x1878 String
p=0x001440ec cb=40 #=1 TID:0x1878
p=0x00142a54 cb=4 #=0 TID:0x1878
total size 552, peak size 560
---------- Leaked Memory Blocks End ------
其中A表示分配 F表示释放
该工具本人初试没有中毒症状,打算纳入个人小宝库中,希望大家喜欢!
下载。
http://www.cppblog.com/wlwlxj/archive/2009/06/03/86660.html
技术回归01-Windows内存分配工具的更多相关文章
- 01 Java 内存分配全面浅析
http://blog.csdn.net/shimiso/article/details/8595564 Java 内存分配全面浅析 本文将由浅入深详细介绍Java内存分配的原理,以帮助新手更轻松的 ...
- windows 内存分配回收检查工具
LeakDiag是微软一款检测memory leak的工具,使用比较简单 首先去下载一个ftp://ftp.microsoft.com/PSS/Tools/Developer%20Support%20 ...
- 【java虚拟机序列】java中的垃圾回收与内存分配策略
在[java虚拟机系列]java虚拟机系列之JVM总述中我们已经详细讲解过java中的内存模型,了解了关于JVM中内存管理的基本知识,接下来本博客将带领大家了解java中的垃圾回收与内存分配策略. 垃 ...
- 内存分配详解 malloc, new, HeapAlloc, VirtualAlloc,GlobalAlloc
很多地方都会使用内存,内存使用过程中操作不当就容易崩溃,无法运行程序,上网Google学习一下,了解整理下他们之间的区别以及使用 ,获益匪浅 0x01 各自的定义和理解 (1)先看GlobalAllo ...
- Netty 中的内存分配浅析-数据容器
本篇接续前一篇继续讲 Netty 中的内存分配.上一篇 先简单做一下回顾: Netty 为了更高效的管理内存,自己实现了一套内存管理的逻辑,借鉴 jemalloc 的思想实现了一套池化内存管理的思路: ...
- C++ Primer 学习笔记_98_特殊的工具和技术 --优化内存分配
特殊的工具和技术 --优化内存分配 引言: C++的内存分配是一种类型化操作:new为特定类型分配内存,并在新分配的内存中构造该类型的一个对象.new表达式自己主动执行合适的构造函数来初始化每一个动态 ...
- 全面介绍Windows内存管理机制及C++内存分配实例(四):内存映射文件
本文背景: 在编程中,很多Windows或C++的内存函数不知道有什么区别,更别谈有效使用:根本的原因是,没有清楚的理解操作系统的内存管理机制,本文企图通过简单的总结描述,结合实例来阐明这个机制. 本 ...
- 全面介绍Windows内存管理机制及C++内存分配实例
转自:http://blog.csdn.net/yeming81/article/details/2046193 本文基本上是windows via c/c++上的内容,笔记做得不错.. 本文背景: ...
- 解决Windows内存问题的两个小工具RamMap和VMMap(这个更牛更好)
来源:http://www.cr173.com/html/13006_1.html .net程序内存监测分配工具(CLR Profiler for .NET Framework 4)官方安装版 类型: ...
随机推荐
- poj 2245 Lotto(dfs)
题目链接:http://poj.org/problem?id=2245 思路分析:无重复元素组合组合问题,使用暴力枚举法,注意剪枝条件. 代码如下: #include <iostream> ...
- linux 命令之sar——监视系统状态
摘要:在进行系统或者内核测试的时候,我们经常需要观察cpu利用率,缓冲区使用情况,文件读写情况等等.在linux系统下,我们可以用sar命令来达到这个要求. sar 命令行的常用格式: sar [op ...
- UVA 10652 Board Wrapping(凸包)
The small sawmill in Mission, British Columbia, hasdeveloped a brand new way of packaging boards for ...
- BZOJ 1455: 罗马游戏( 配对堆 + 并查集 )
可并堆水题 --------------------------------------------------------- #include<bits/stdc++.h> usin ...
- nginx+apache 404错误页面
公司新系统 随风做的 给客户演示出错不想让客户看到 自动返回上一页面. 刚开始按照网上说的 在nginx 处理: # 定义错误提示页面 error_page 500 502 503 504 / ...
- [转]CentO下限制SSH登录次数
应公司内部网站等级测评的需求,正逐渐加强系统安全防护. 设备默认 3 次验证失败自动退出,并且结束会话:网络登录连接超时自动退出时间 5 分钟: 第一种方法:已验证. 1.ssh超时时间设置 # cd ...
- java--String方法
String : 字符串类型 一.构造函数 String(byte[ ] bytes):通过byte数组构造字符串对象. String(char[ ] value):通过char数组构造字符串对象. ...
- if…else…if…else…
参见以前做过的练习一元二次方程 #include <stdio.h> #include <math.h> /* 一元二次方程的标准形式:ax2+bx+c=0 a,b,c为常数, ...
- CodeForces 22B Bargaining Table 简单DP
题目很好理解,问你的是在所给的图中周长最长的矩形是多长嗯用坐标(x1, y1, x2, y2)表示一个矩形,暴力图中所有矩形易得递推式:(x1, y1, x2, y2)为矩形的充要条件为: (x1, ...
- UltraEdit for mac 3.2.0.10免费破解版下载!!
http://www.mactech.cn/a/108.html UltraEdit for mac 3.2.0.10破解版下载地址 看很多朋友不知道算号器的使用方法,分享如下: 1. 解压Ultra ...