[VS debug] C++ new做的工作

int _tmain(int argc, _TCHAR* argv[])
{
	int *pData = new int;
}

首先会调用new.cpp中的void *__CRTDECL operator new(size_t size)函数随后跳转到malloc -> _nh_malloc_dbg -> _nh_malloc_dbg_impl -> _heap_alloc_dbg_impl ,在

_heap_alloc_dbg_impl函数中做了一些额外的动作:

   1. 分配的大小

             

size = sizeof(_CrtMemBlockHeader) + nSize + nNoMansLandSize;

        其中_CrtMemBlockHeader结构体如下:

#define nNoMansLandSize 4

typedef struct _CrtMemBlockHeader
{
        struct _CrtMemBlockHeader * pBlockHeaderNext;
        struct _CrtMemBlockHeader * pBlockHeaderPrev;
        char *                      szFileName;
        int                         nLine;
#ifdef _WIN64
        /* These items are reversed on Win64 to eliminate gaps in the struct
         * and ensure that sizeof(struct)%16 == 0, so 16-byte alignment is
         * maintained in the debug heap.
         */
        int                         nBlockUse;
        size_t                      nDataSize;
#else  /* _WIN64 */
        size_t                      nDataSize;
        int                         nBlockUse;
#endif  /* _WIN64 */
        long                        lRequest;
        unsigned char               gap[nNoMansLandSize];
        /* followed by:
         *  unsigned char           data[nDataSize];
         *  unsigned char           anotherGap[nNoMansLandSize];
         */
} _CrtMemBlockHeader;

由于nSize=sizeof(int)=4,所有最终分配的大小为size = 32+4+4 = 40字节;


2. 节点信息的填充

_CrtMemBlockHeader 是一个双向链表的节点,节点中的信息为dbg调试使用,当申请完40字节后则开始对_CrtMemBlockHeader信息进行填充,pBlockHeaderNext指向上一次分配内存的那个节点,pBlockHeaderPrev指向NULL表示自己是整个链表的首节点。通过这两个字段赋值可知道整个链表是按照分配的时间排下来的。

_CrtMemBlockHeader 的最后一个字段是unsigned char gap[4],并且在分配内存大小时

size = sizeof(_CrtMemBlockHeader) + nSize + nNoMansLandSize;
最后也多分配了nNoMansLandSize即4个字节,这是为什么呢?

这两个小缓冲区是为了检测内存操作越界而创建的,debug下这两个缓冲区都初始化为0xFD的值。如图:



用户buffer会填充0xCD,两边的缓冲区会填充0xFD。所有信息填充完毕后,返回Pointer作为返回值。

不是很了解可以在debug环境下进行单步调试查看。


参考资料: https://msdn.microsoft.com/zh-cn/library/974tc9t1.aspx

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在给定的代码片段,"#define new DEBUG_NEW"是一个预处理器指令,它将new关键字重新定义为DEBUG_NEW。这个重新定义的目的是帮助在调试过程查找内存错误。在调试模式下,使用DEBUG_NEW替代new运算符来分配对象,它会记录分配对象的文件名和行号。当使用CMemoryState::DumpAllObjectSince函数时,可以显示每个使用DEBUG_NEW分配的对象的文件名和行号信息。这个重新定义可以通过在用户的资源文件插入"#define new DEBUG_NEW"指令来启用。在发布版本DEBUG_NEW将简单地进行标准的new操作,不会产生文件名和行号信息。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [C++关于Crt的内存泄漏检测的分析介绍](https://download.csdn.net/download/weixin_38723242/14871457)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [#ifdef _DEBUG #define new DEBUG_NEW #undef THIS_FILE static char THIS_FILE[] = __FILE__; #endif 语句...](https://blog.csdn.net/Bason09/article/details/8793481)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值