GlobalAlloc,GlobalLock,LockResource等用法

转自:

https://blog.csdn.net/zxm342698145/article/details/26734049

GlobalAlloc 

函数原型
GlobalAlloc(UINT uFLAG,DWORD dwBytes);
说明 
分配一个全局内存块 
返回值 
Long,返回全局内存句柄。零表示失败。会设置GetLastError 
参数表 
参数 类型及说明 
uFlags Long,对分配的内存类型进行定义的常数标志,如下所示: 
             GMEM_FIXED 分配一个固定内存块 
             GMEM_MOVEABLE 分配一个可移动内存块 
             GMEM_DISCARDABLE 分配一个可丢弃内存块 
             GMEM_NOCOMPACT 堆在这个函数调用期间不进行累积 
             GMEM_NODISCARD 函数调用期间不丢弃任何内存块 
             GMEM_ZEROINIT 新分配的内存块全部初始化成零 
dwBytes Long,要分配的字符数 
注解  
如指定了 GMEM_FIXED,那么返回值就是要使用的实际内存地址即指针(GlobalLock 会返回同样的值)——所以在使用固定内存块的时候不需要执行一个 GlobalLock/GlobalUnlock 操作
由于 Win32 采用了高级的内存管理方案,所以使用可移动的内存块并没有什么好处
用这个函数分配的内存块允许在8位边界以内

GlobalLock  
函数功能描述
:锁定一个全局的内存对象,返回指向该对象的第一个字节的指针
函数原型
LPVOID GlobalLock( HGLOBAL hMem )
参数
hMem:全局内存对象的句柄。这个句柄是通过GlobalAlloc或GlobalReAlloc来得到的
返回值:
调用成功,返回指向该对象的第一个字节的指针
调用失败,返回NULL,可以用GetLastError来获得出错信息
注意:
调用过GlobalLock锁定一块内存区后,一定要调用GlobalUnlock来解锁。

GlobalUnlock
函数功能描述:
解除被锁定的全局内存对象
函数原型:BOOL GlobalUnlock( HGLOBAL hMem );
参数:hMem:全局内存对象的句柄
返回值:
非零值,指定的内存对象仍处于被锁定状态
0,函数执行出错,可以用GetLastError来获得出错信息,如果返回NO_ERROR,则表示内存对象已经解锁了
注意:    这个函数实际上是将内存对象的锁定计数器减一,如果计数器不为0,则表示执行过多个GlobalLock函数来对这个内存对象加锁,需要对应数目的GlobalUnlock函数来解锁。
    如果通过GetLastError函数返回错误码为ERROR_NOT_LOCKED,则表示未加锁或已经解锁。

 

调用GlobalAlloc函数分配一块内存,该函数会返回分配的内存句柄。 

 GlobalAlloc申请的内存分两种,一种是GMEM_FIXED,另一种是GMEM_MOVEABLE。两者的差别只要在于

GMEM_MOVEABLE类型的内存操作系统是可以移动的,比如堆中有好几块小内存,当再申请一大块内存时,操作系统会移动GMEM_MOVEABLE类型的内存来合并出一大块。正因为GMEM_MOVEABLE是可移动的,所以要用句柄标识,不能用内存地址标识,在使用时通过GlobalLock由句柄得到内存地址

对于GMEM_FIXED类型的,该函数返回的句柄就是内存指针,可以直接当内存指针使用


调用GlobalLock函数锁定内存块,该函数接受一个内存句柄作为参数,然后返回一个指向被锁定的内存块的指针。 

您可以用该指针来读写内存。


调用GlobalUnlock函数来解锁先前被锁定的内存,该函数使得指向内存块的指针无效。 
调用GlobalFree函数来释放内存块。您必须传给该函数一个内存句柄。

比如将一个图片(位图、png什么的都可以)的二进制转换为程序可以用的资源:

 

//参数分别是:资源二进制流(把一张图片用可以查看16进制文本打开,拷贝就行);字节流大小;要转换为可以用的资源
 
  1. BOOL ImageFromIDResource( unsigned char * lpResStreamName,unsigned int len,Gdiplus::Image * &pImg )

  2. {

  3. // Allocate global memory on which to create stream

  4. HGLOBAL hMem = GlobalAlloc(GMEM_FIXED, len) ;

  5. //BYTE* pmem = (BYTE*)GlobalLock(hMem); //根据前面解释,由于参数是GMEM_FIXED可以不用调用该函数

  6. //memcpy(pmem, lpResStreamName,len); //同上

  7. memcpy(hMem, lpResStreamName, len) ;

  8. IStream* pstm;

  9. CreateStreamOnHGlobal(hMem,FALSE,&pstm) ;

  10. // load from stream

  11. pImg = Gdiplus::Image::FromStream(pstm);

  12. // free/release stuff

  13. pstm->Release();

  14. //GlobalUnlock(m_hMem); //对应的也不需啊

  15. return TRUE;

  16. }

将VC导入的资源又如何转为临时文件呢?需要用到LockResource函数,下面给出代码,相信不难分析。

  1. BOOL ReleaseResourceToTempPath(int iResId, LPTSTR lpType, LPTSTR lpFileName)

  2. {

  3. DWORD dwSize;

  4. HGLOBAL hGlobal;

  5. LPVOID lpBuffer;

  6. HRSRC hRes = ::FindResource(NULL, MAKEINTRESOURCE(iResId), lpType);

  7. if (hRes){

  8. hGlobal = ::LoadResource(NULL, hRes);

  9. dwSize = ::SizeofResource(NULL, hRes);

  10. if (hGlobal && dwSize){

  11. lpBuffer = ::LockResource(hGlobal);

  12. TCHAR pTempFileName[256] ;

  13. if (lpBuffer){

  14. GetTempPath(MAX_PATH, pTempFileName);

  15. strcat_s(pTempFileName, sizeof(pTempFileName), lpFileName);

  16. FILE* fp;

  17. fopen_s(&fp, pTempFileName, _T("wb"));

  18. if (dwSize == fwrite(lpBuffer, sizeof(char), dwSize, fp)){

  19. fclose(fp);

  20. return TRUE;

  21. }

  22. else{

  23. fclose(fp);

  24. }

  25. }

  26. }

  27. }

  28. return FALSE;

  29. }

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值