VS_FIXEDFILEINFO结构包含了文件的版本信息:
typedef struct tagVS_FIXEDFILEINFO { DWORD dwSignature; //包含的值是0xFEEF04BD DWORD dwStrucVersion; //该结构的32位二进制版本号,高16位是主版本号,低16位是副版本号 DWORD dwFileVersionMS; //该文件二进制版本号的高32bits DWORD dwFileVersionLS; //该文件二进制版本号的低32bits DWORD dwProductVersionMS; //发布该文件的产品二进制版本号高32bits DWORD dwProductVersionLS; //发布该文件的产品二进制版本号低32bits DWORD dwFileFlagsMask; //比特掩码,标志dwFileFlags的有效位 DWORD dwFileFlags; //VS_FF_DEBUG---该文件包含调试信息或是由调试版编译的 //VS_FF_INFOINFERRED---文件的版本结构是动态创建的, //因此,该结构中有的成员是空的或不正确的 //VS_FF_PATCHED---该文件被修改过 //VS_FF_PRERELEASE---该文件是开发版,不是商业发布版 //VS_FF_PRIVATEBUILD---该文件不是由标准发布步骤构建的 //VS_FF_SPECIALBUILD---该文件是由标准发布步骤构建的, //但是相同版本号文件的变种 DWORD dwFileOS; //该文件设计用于的操作系统 DWORD dwFileType; //文件类型:VFT_APP---文件包含一个应用程序 VFT_DLL---文件包含一个DLL VFT_DRV---文件包含一个设备驱动 VFT_FONT---文件包含一个字体文件 VFT_STATIC_LIB---文件包含一个静态链接库 VFT_UNKNOWN---文件类型未知 VFT_VXD---文件包含一个虚拟设备 DWORD dwFileSubtype; //文件的子类型,由dwFileType决定 DWORD dwFileDateMS; //二进制文件创建日期和时间戳的高32bits DWORD dwFileDateLS; //二进制文件创建日期和时间戳的低32bits } VS_FIXEDFILEINFO;
GetFileVersionInfoSize函数用于判断系统能否检索到指定文件的版本信息,如果可以,函数返回版本信息的字节大小:
DWORD WINAPI GetFileVersionInfoSize( __in LPCTSTR lptstrFilename, //指定文件的名称 __out_opt LPDWORD lpdwHandle //一个变量的指针,该函数将该变量设为0 );
GetFileVersionInfo函数用来获得指定文件的版本信息:
BOOL WINAPI GetFileVersionInfo( __in LPCTSTR lptstrFilename, //文件名 __reserved DWORD dwHandle, //保留值 __in DWORD dwLen, //lpData指向缓冲区的大小,使用函数GetFileVersionInfoSize得到 __out LPVOID lpData //指向存放文件版本信息的缓冲区的指针 );
VerQueryValue函数用于从指定的版本信息源获取版本信息,在调用该函数之前,需要先依次调用函数GetFileVersionInfoSize和GetFileVersionInfo:
BOOL WINAPI VerQueryValue( __in LPCVOID pBlock, //由函数GetFileVersionInfo得到的版本信息源 __in LPCTSTR lpSubBlock, //“/”表示该函数获取VS_FIXEDFILEINFO结构信息 //为“/VarFileInfo/Translation”表示该函数获取文件的翻译表 //为“/StringFileInfo/lang-codepage/string-name”表示该函数获取文件的字符串信息 __out LPVOID *lplpBuffer, //返回指向pBlock指向的地址,当pBlock被释放时,该参数也被释放 __out PUINT puLen //lplpBuffer指向的数据的字节大小 );
上面参数lpSubBlock取值中的string-name必须是下面系统预定义的字符串之一:
下面代码实例封装了一个文件版本信息类,使用上面介绍的函数方便地获取文件版本信息,头文件定义如下FileVersion.h:
// FileVersion.h: interface for the CFileVersion class. // by Manuel Laflamme ////////////////////////////////////////////////////////////////////// #ifndef __FILEVERSION_H_ #define __FILEVERSION_H_ #if _MSC_VER >= 1000 #pragma once #endif // _MSC_VER >= 1000 class CFileVersion { // Construction public: CFileVersion(); // Operations public: BOOL Open(LPCTSTR lpszModuleName); void Close(); CString QueryValue(LPCTSTR lpszValueName, DWORD dwLangCharset = 0); CString GetFileDescription() {return QueryValue(_T("FileDescription")); }; CString GetFileVersion() {return QueryValue(_T("FileVersion")); }; CString GetInternalName() {return QueryValue(_T("InternalName")); }; CString GetCompanyName() {return QueryValue(_T("CompanyName")); }; CString GetLegalCopyright() {return QueryValue(_T("LegalCopyright")); }; CString GetOriginalFilename() {return QueryValue(_T("OriginalFilename"));}; CString GetProductName() {return QueryValue(_T("ProductName")); }; CString GetProductVersion() {return QueryValue(_T("ProductVersion")); }; BOOL GetFixedInfo(VS_FIXEDFILEINFO& vsffi); CString GetFixedFileVersion(); CString GetFixedProductVersion(); // Attributes protected: LPBYTE m_lpVersionData; DWORD m_dwLangCharset; // Implementation public: ~CFileVersion(); }; #endif // __FILEVERSION_H_
头文件的实现如下FileVersion.cpp:
// FileVersion.cpp: implementation of the CFileVersion class. // by Manuel Laflamme ////////////////////////////////////////////////////////////////////// #include "FileVersion.h" #pragma comment(lib, "version") #ifdef _DEBUG #undef THIS_FILE static char THIS_FILE[]=__FILE__; #define new DEBUG_NEW #endif ////////////////////////////////////////////////////////////////////// CFileVersion::CFileVersion() { m_lpVersionData = NULL; m_dwLangCharset = 0; } CFileVersion::~CFileVersion() { Close(); } void CFileVersion::Close() { delete[] m_lpVersionData; m_lpVersionData = NULL; m_dwLangCharset = 0; } BOOL CFileVersion::Open(LPCTSTR lpszModuleName) { ASSERT(_tcslen(lpszModuleName) > 0); ASSERT(m_lpVersionData == NULL); // Get the version information size for allocate the buffer DWORD dwHandle; DWORD dwDataSize = ::GetFileVersionInfoSize((LPTSTR)lpszModuleName, &dwHandle); if ( dwDataSize == 0 ) return FALSE; // Allocate buffer and retrieve version information m_lpVersionData = new BYTE[dwDataSize]; if (!::GetFileVersionInfo((LPTSTR)lpszModuleName, dwHandle, dwDataSize, (void**)m_lpVersionData) ) { Close(); return FALSE; } // Retrieve the first language and character-set identifier UINT nQuerySize; DWORD* pTransTable; if (!::VerQueryValue(m_lpVersionData, _T("//VarFileInfo//Translation"), (void **)&pTransTable, &nQuerySize) ) { Close(); return FALSE; } // Swap the words to have lang-charset in the correct format m_dwLangCharset = MAKELONG(HIWORD(pTransTable[0]), LOWORD(pTransTable[0])); return TRUE; } CString CFileVersion::QueryValue(LPCTSTR lpszValueName, DWORD dwLangCharset /* = 0*/) { // Must call Open() first ASSERT(m_lpVersionData != NULL); if ( m_lpVersionData == NULL ) return (CString)_T(""); // If no lang-charset specified use default if ( dwLangCharset == 0 ) dwLangCharset = m_dwLangCharset; // Query version information value UINT nQuerySize; LPVOID lpData; CString strValue, strBlockName; strBlockName.Format(_T("//StringFileInfo//%08lx//%s"), dwLangCharset, lpszValueName); if ( ::VerQueryValue((void **)m_lpVersionData, strBlockName.GetBuffer(0), &lpData, &nQuerySize) ) strValue = (LPCTSTR)lpData; strBlockName.ReleaseBuffer(); return strValue; } BOOL CFileVersion::GetFixedInfo(VS_FIXEDFILEINFO& vsffi) { // Must call Open() first ASSERT(m_lpVersionData != NULL); if ( m_lpVersionData == NULL ) return FALSE; UINT nQuerySize; VS_FIXEDFILEINFO* pVsffi; if ( ::VerQueryValue((void **)m_lpVersionData, _T("//"), (void**)&pVsffi, &nQuerySize) ) { vsffi = *pVsffi; return TRUE; } return FALSE; } CString CFileVersion::GetFixedFileVersion() { CString strVersion; VS_FIXEDFILEINFO vsffi; if ( GetFixedInfo(vsffi) ) { strVersion.Format (_T("%u,%u,%u,%u"),HIWORD(vsffi.dwFileVersionMS), LOWORD(vsffi.dwFileVersionMS), HIWORD(vsffi.dwFileVersionLS), LOWORD(vsffi.dwFileVersionLS)); } return strVersion; } CString CFileVersion::GetFixedProductVersion() { CString strVersion; VS_FIXEDFILEINFO vsffi; if ( GetFixedInfo(vsffi) ) { strVersion.Format (_T("%u,%u,%u,%u"), HIWORD(vsffi.dwProductVersionMS), LOWORD(vsffi.dwProductVersionMS), HIWORD(vsffi.dwProductVersionLS), LOWORD(vsffi.dwProductVersionLS)); } return strVersion; }