Winapi 如何从资源中获取版本信息?

Winapi 如何从资源中获取版本信息?,winapi,visual-c++,resources,Winapi,Visual C++,Resources,我在参考资料中声明了版本信息: 100 VERSIONINFO FILEVERSION 1,0,0,2 PRODUCTVERSION 1,0,0,2 FILEOS VOS_NT FILETYPE VFT_APP { BLOCK "StringFileInfo" { BLOCK "000004b0" { VALUE "FileDescription", "My application" VALUE "F

我在参考资料中声明了版本信息:

100 VERSIONINFO
FILEVERSION 1,0,0,2
PRODUCTVERSION 1,0,0,2
FILEOS VOS_NT
FILETYPE VFT_APP
{
    BLOCK "StringFileInfo"
    {
        BLOCK "000004b0"
        {
            VALUE "FileDescription", "My application"
            VALUE "FileVersion", "1.0.0.2"
            VALUE "InternalName", "app.exe"
            VALUE "LegalCopyright", "Copyright ©  2012 by David."
            VALUE "OriginalFilename", "app.exe"
            VALUE "ProductName", "app"
            VALUE "ProductVersion", "1.0.0.2"
            VALUE "Assembly Version", "1.0.0.2"
        }
    }

    BLOCK "VarFileInfo"
    {
        VALUE "Translation", 0x0000 0x04B0
    }
}
我通过以下方式获取版本信息:

HRSRC hResInfo;
HGLOBAL hResData;
LPCVOID pRes;
UINT uLen;
VS_FIXEDFILEINFO *lpFfi;

hResInfo = FindResource(hInst, MAKEINTRESOURCE(100), RT_VERSION);
MessageBox(0, "FindResource", 0,0);

hResData = LoadResource(hInst, hResInfo);
MessageBox(0, "LoadResource", 0,0);

pRes = LockResource(hResData);
MessageBox(0, "LockResource", 0,0);

VerQueryValue(pRes, "\\" ,(LPVOID*)&lpFfi, &uLen);
MessageBox(0, "VerQueryValue", 0,0);

FreeResource(hResData);

DWORD dwFileVersionMS = lpFfi->dwFileVersionMS;
DWORD dwFileVersionLS = lpFfi->dwFileVersionLS;

DWORD dwLeftMost     = HIWORD(dwFileVersionMS);
DWORD dwSecondLeft   = LOWORD(dwFileVersionMS);
DWORD dwSecondRight  = HIWORD(dwFileVersionLS);
DWORD dwRightMost    = LOWORD(dwFileVersionLS);

函数ValQueCalk是一个错误,因为程序中断(带有文本的消息框“ValQuealValk”没有显示),VisualC++显示了以下消息:

ProxyCU.exe中0x77bf15a5处的首次机会异常:0xC0000005: 访问冲突写入位置0x00483192

如何修复此代码

关于,David

VerQueryValue()
无法直接从原始资源访问版本信息。您必须在内存中复制资源,然后将该内存传递给
VerQueryValue()
。原因是
VerQueryValue()
设计用于使用
GetFileVersionInfo()
,这需要用户分配的可写内存块,并在该内存中执行某些修复。访问
VS_FIXEDFILEINFO
结构不需要修复,但内存块仍必须是可写的。无法将原始资源直接传递给
VerQueryValue()
,因为它是只读内存

请尝试以下方法:

HRSRC hResInfo;
DWORD dwSize;
HGLOBAL hResData;
LPVOID pRes, pResCopy;
UINT uLen;
VS_FIXEDFILEINFO *lpFfi;

hResInfo = FindResource(hInst, MAKEINTRESOURCE(100), RT_VERSION);
dwSize = SizeofResource(hInst, hResInfo);
hResData = LoadResource(hInst, hResInfo);
pRes = LockResource(hResData);
pResCopy = LocalAlloc(LMEM_FIXED, dwSize);
CopyMemory(pResCopy, pRes, dwSize);
FreeResource(hResData);

VerQueryValue(pResCopy, TEXT("\\"), (LPVOID*)&lpFfi, &uLen);

DWORD dwFileVersionMS = lpFfi->dwFileVersionMS;
DWORD dwFileVersionLS = lpFfi->dwFileVersionLS;

DWORD dwLeftMost     = HIWORD(dwFileVersionMS);
DWORD dwSecondLeft   = LOWORD(dwFileVersionMS);
DWORD dwSecondRight  = HIWORD(dwFileVersionLS);
DWORD dwRightMost    = LOWORD(dwFileVersionLS);

LocalFree(pResCopy);
更新:仅当您访问
VS\u FIXEDFILEINFO
结构时,此选项才有效。如果需要访问任何其他值,则必须使用
GetFileVersionInfo()
。根据陈雷蒙的博客:

文档说明VerQueryValue的第一个参数必须是GetFileVersionInfo函数返回的缓冲区,这是有原因的GetFileVersionInfo返回的缓冲区是一个不透明的数据块,专门格式化,以便VerQueryValue可以工作。您不应该查看缓冲区内部,当然也不能尝试“以其他方式获取数据”。因为如果这样做,VerQueryValue将在缓冲区中查找未按函数期望的方式格式化的内容。


上述操作将在调试模式下创建堆损坏错误消息,例如“释放后在b753ed4修改的释放堆块b753e70”。许多年前,有人在上发布了此问题。今天还在下雨。通过使dwSize足够大,例如将其乘以4,可以使消息消失

我通常会考虑到版本信息是在哪里定义的,所以当在代码中使用它时,它是编译器定义的,而不是调用
LoadResource()
。如果你不能让你的代码工作,你可以认为它是一个选项。我建议在LooRealCype()之后检查错误代码(在调试器中@ Err监视,或者GETLaSTError)。此外,请检查此地址0x00483192是否位于pRes指出的内存中。与问题完全无关,但为什么您的版本资源id为100?据我所知,它预计将在id 1,这是VS_版本_信息+1,我只是需要和使用这个。非常感谢。一个很小的注释:
freereresource
调用是不需要的,并且根据文档返回
FALSE
。是否有任何官方文档说明您必须复制该块?的文档建议不应该调用它。版本资源不一定是100,可以是任意数字。在我正在处理的遗留项目中,它是
MAKEINTRESOURCE(1)
@Soonts true,但在OP给出的示例中,它是100。