《Windows API巡礼》---GetFileVersionInfo函数和VerQueryValue函数

2024-04-24 22:32

本文主要是介绍《Windows API巡礼》---GetFileVersionInfo函数和VerQueryValue函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

VS_FIXEDFILEINFO结构包含了文件的版本信息:

  1. typedef struct tagVS_FIXEDFILEINFO {  
  2.   DWORD dwSignature; //包含的值是0xFEEF04BD   
  3.   DWORD dwStrucVersion; //该结构的32位二进制版本号,高16位是主版本号,低16位是副版本号   
  4.   DWORD dwFileVersionMS; //该文件二进制版本号的高32bits   
  5.   DWORD dwFileVersionLS; //该文件二进制版本号的低32bits   
  6.   DWORD dwProductVersionMS; //发布该文件的产品二进制版本号高32bits   
  7.   DWORD dwProductVersionLS; //发布该文件的产品二进制版本号低32bits   
  8.   DWORD dwFileFlagsMask; //比特掩码,标志dwFileFlags的有效位   
  9.   DWORD dwFileFlags; //VS_FF_DEBUG---该文件包含调试信息或是由调试版编译的   
  10.                              //VS_FF_INFOINFERRED---文件的版本结构是动态创建的,   
  11. //因此,该结构中有的成员是空的或不正确的   
  12.                              //VS_FF_PATCHED---该文件被修改过   
  13.                              //VS_FF_PRERELEASE---该文件是开发版,不是商业发布版   
  14.                              //VS_FF_PRIVATEBUILD---该文件不是由标准发布步骤构建的   
  15.                              //VS_FF_SPECIALBUILD---该文件是由标准发布步骤构建的,   
  16. //但是相同版本号文件的变种   
  17.   DWORD dwFileOS; //该文件设计用于的操作系统   
  18.   DWORD dwFileType; //文件类型:VFT_APP---文件包含一个应用程序   
  19.                                            VFT_DLL---文件包含一个DLL  
  20.                                            VFT_DRV---文件包含一个设备驱动  
  21.                                            VFT_FONT---文件包含一个字体文件  
  22.                                            VFT_STATIC_LIB---文件包含一个静态链接库  
  23.                                            VFT_UNKNOWN---文件类型未知  
  24.                                            VFT_VXD---文件包含一个虚拟设备  
  25.   DWORD dwFileSubtype; //文件的子类型,由dwFileType决定   
  26.   DWORD dwFileDateMS; //二进制文件创建日期和时间戳的高32bits   
  27.   DWORD dwFileDateLS; //二进制文件创建日期和时间戳的低32bits   
  28. } VS_FIXEDFILEINFO;  

GetFileVersionInfoSize函数用于判断系统能否检索到指定文件的版本信息,如果可以,函数返回版本信息的字节大小:

  1. DWORD WINAPI GetFileVersionInfoSize(  
  2.   __in       LPCTSTR lptstrFilename, //指定文件的名称   
  3.   __out_opt  LPDWORD lpdwHandle //一个变量的指针,该函数将该变量设为0   
  4. );  

GetFileVersionInfo函数用来获得指定文件的版本信息:

  1. BOOL WINAPI GetFileVersionInfo(  
  2.   __in        LPCTSTR lptstrFilename, //文件名   
  3.   __reserved  DWORD dwHandle, //保留值   
  4.   __in        DWORD dwLen, //lpData指向缓冲区的大小,使用函数GetFileVersionInfoSize得到   
  5.   __out       LPVOID lpData //指向存放文件版本信息的缓冲区的指针   
  6. );  

VerQueryValue函数用于从指定的版本信息源获取版本信息,在调用该函数之前,需要先依次调用函数GetFileVersionInfoSizeGetFileVersionInfo

上面参数lpSubBlock取值中的string-name必须是下面系统预定义的字符串之一:

下面代码实例封装了一个文件版本信息类,使用上面介绍的函数方便地获取文件版本信息,头文件定义如下FileVersion.h: 

  1. // FileVersion.h: interface for the CFileVersion class.   
  2. // by Manuel Laflamme   
  3. //   
  4. #ifndef __FILEVERSION_H_   
  5. #define __FILEVERSION_H_   
  6. #if _MSC_VER >= 1000   
  7. #pragma once   
  8. #endif // _MSC_VER >= 1000   
  9. class CFileVersion  
  10. {   
  11. // Construction   
  12. public:   
  13.     CFileVersion();  
  14. // Operations      
  15. public:   
  16.     BOOL    Open(LPCTSTR lpszModuleName);  
  17.     void    Close();  
  18.     CString QueryValue(LPCTSTR lpszValueName, DWORD dwLangCharset = 0);  
  19.     CString GetFileDescription()  {return QueryValue(_T("FileDescription")); };  
  20.     CString GetFileVersion()      {return QueryValue(_T("FileVersion"));     };  
  21.     CString GetInternalName()     {return QueryValue(_T("InternalName"));    };  
  22.     CString GetCompanyName()      {return QueryValue(_T("CompanyName"));     };   
  23.     CString GetLegalCopyright()   {return QueryValue(_T("LegalCopyright"));  };  
  24.     CString GetOriginalFilename() {return QueryValue(_T("OriginalFilename"));};  
  25.     CString GetProductName()      {return QueryValue(_T("ProductName"));     };  
  26.     CString GetProductVersion()   {return QueryValue(_T("ProductVersion"));  };  
  27.     BOOL    GetFixedInfo(VS_FIXEDFILEINFO& vsffi);  
  28.     CString GetFixedFileVersion();  
  29.     CString GetFixedProductVersion();  
  30. // Attributes   
  31. protected:  
  32.     LPBYTE  m_lpVersionData;   
  33.     DWORD   m_dwLangCharset;   
  34. // Implementation   
  35. public:  
  36.     ~CFileVersion();   
  37. };   
  38. #endif  // __FILEVERSION_H_  

头文件的实现如下FileVersion.cpp:

  1. // FileVersion.cpp: implementation of the CFileVersion class.   
  2. // by Manuel Laflamme    
  3. //   
  4. #include "FileVersion.h"   
  5. #pragma comment(lib, "version")   
  6. #ifdef _DEBUG   
  7. #undef THIS_FILE   
  8. static char THIS_FILE[]=__FILE__;  
  9. #define new DEBUG_NEW   
  10. #endif   
  11. //   
  12. CFileVersion::CFileVersion()   
  13. {   
  14.     m_lpVersionData = NULL;  
  15.     m_dwLangCharset = 0;  
  16. }  
  17. CFileVersion::~CFileVersion()   
  18. {   
  19.     Close();  
  20. }   
  21. void CFileVersion::Close()  
  22. {  
  23.     delete[] m_lpVersionData;   
  24.     m_lpVersionData = NULL;  
  25.     m_dwLangCharset = 0;  
  26. }  
  27. BOOL CFileVersion::Open(LPCTSTR lpszModuleName)  
  28. {  
  29.     ASSERT(_tcslen(lpszModuleName) > 0);  
  30.     ASSERT(m_lpVersionData == NULL);  
  31.     // Get the version information size for allocate the buffer   
  32.     DWORD dwHandle;       
  33.     DWORD dwDataSize = ::GetFileVersionInfoSize((LPTSTR)lpszModuleName, &dwHandle);   
  34.     if ( dwDataSize == 0 )   
  35.         return FALSE;  
  36.     // Allocate buffer and retrieve version information   
  37.     m_lpVersionData = new BYTE[dwDataSize];   
  38.     if (!::GetFileVersionInfo((LPTSTR)lpszModuleName, dwHandle, dwDataSize,   
  39.                               (void**)m_lpVersionData) )  
  40.     {  
  41.         Close();  
  42.         return FALSE;  
  43.     }  
  44.     // Retrieve the first language and character-set identifier   
  45.     UINT nQuerySize;  
  46.     DWORD* pTransTable;  
  47.     if (!::VerQueryValue(m_lpVersionData, _T("\\VarFileInfo\\Translation"),  
  48.                          (void **)&pTransTable, &nQuerySize) )  
  49.     {  
  50.         Close();  
  51.         return FALSE;  
  52.     }  
  53.     // Swap the words to have lang-charset in the correct format   
  54.     m_dwLangCharset = MAKELONG(HIWORD(pTransTable[0]), LOWORD(pTransTable[0]));  
  55.     return TRUE;  
  56. }  
  57. CString CFileVersion::QueryValue(LPCTSTR lpszValueName,   
  58.                                  DWORD dwLangCharset /* = 0*/)  
  59. {  
  60.     // Must call Open() first   
  61.     ASSERT(m_lpVersionData != NULL);  
  62.     if ( m_lpVersionData == NULL )  
  63.         return (CString)_T("");  
  64.     // If no lang-charset specified use default   
  65.     if ( dwLangCharset == 0 )  
  66.         dwLangCharset = m_dwLangCharset;  
  67.     // Query version information value   
  68.     UINT nQuerySize;  
  69.     LPVOID lpData;  
  70.     CString strValue, strBlockName;  
  71.     strBlockName.Format(_T("\\StringFileInfo\\%08lx\\%s"),   
  72.                          dwLangCharset, lpszValueName);  
  73.     if ( ::VerQueryValue((void **)m_lpVersionData, strBlockName.GetBuffer(0),   
  74.                          &lpData, &nQuerySize) )  
  75.         strValue = (LPCTSTR)lpData;  
  76.     strBlockName.ReleaseBuffer();  
  77.     return strValue;  
  78. }  
  79. BOOL CFileVersion::GetFixedInfo(VS_FIXEDFILEINFO& vsffi)  
  80. {  
  81.     // Must call Open() first   
  82.     ASSERT(m_lpVersionData != NULL);  
  83.     if ( m_lpVersionData == NULL )  
  84.         return FALSE;  
  85.     UINT nQuerySize;  
  86.     VS_FIXEDFILEINFO* pVsffi;  
  87.     if ( ::VerQueryValue((void **)m_lpVersionData, _T("//"),  
  88.                          (void**)&pVsffi, &nQuerySize) )  
  89.     {  
  90.         vsffi = *pVsffi;  
  91.         return TRUE;  
  92.     }  
  93.     return FALSE;  
  94. }  
  95. CString CFileVersion::GetFixedFileVersion()  
  96. {  
  97.     CString strVersion;  
  98.     VS_FIXEDFILEINFO vsffi;  
  99.     if ( GetFixedInfo(vsffi) )  
  100.     {  
  101.         strVersion.Format (_T("%u,%u,%u,%u"),HIWORD(vsffi.dwFileVersionMS),  
  102.             LOWORD(vsffi.dwFileVersionMS),  
  103.             HIWORD(vsffi.dwFileVersionLS),  
  104.             LOWORD(vsffi.dwFileVersionLS));  
  105.     }  
  106.     return strVersion;  
  107. }  
  108. CString CFileVersion::GetFixedProductVersion()  
  109. {  
  110.     CString strVersion;  
  111.     VS_FIXEDFILEINFO vsffi;  
  112.     if ( GetFixedInfo(vsffi) )  
  113.     {  
  114.         strVersion.Format (_T("%u,%u,%u,%u"), HIWORD(vsffi.dwProductVersionMS),  
  115.             LOWORD(vsffi.dwProductVersionMS),  
  116.             HIWORD(vsffi.dwProductVersionLS),  
  117.             LOWORD(vsffi.dwProductVersionLS));  
  118.     }  
  119.     return strVersion;  
  120. }  

这篇关于《Windows API巡礼》---GetFileVersionInfo函数和VerQueryValue函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/933017

相关文章

PostgreSQL中rank()窗口函数实用指南与示例

《PostgreSQL中rank()窗口函数实用指南与示例》在数据分析和数据库管理中,经常需要对数据进行排名操作,PostgreSQL提供了强大的窗口函数rank(),可以方便地对结果集中的行进行排名... 目录一、rank()函数简介二、基础示例:部门内员工薪资排名示例数据排名查询三、高级应用示例1. 每

全面掌握 SQL 中的 DATEDIFF函数及用法最佳实践

《全面掌握SQL中的DATEDIFF函数及用法最佳实践》本文解析DATEDIFF在不同数据库中的差异,强调其边界计算原理,探讨应用场景及陷阱,推荐根据需求选择TIMESTAMPDIFF或inte... 目录1. 核心概念:DATEDIFF 究竟在计算什么?2. 主流数据库中的 DATEDIFF 实现2.1

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串

Knife4j+Axios+Redis前后端分离架构下的 API 管理与会话方案(最新推荐)

《Knife4j+Axios+Redis前后端分离架构下的API管理与会话方案(最新推荐)》本文主要介绍了Swagger与Knife4j的配置要点、前后端对接方法以及分布式Session实现原理,... 目录一、Swagger 与 Knife4j 的深度理解及配置要点Knife4j 配置关键要点1.Spri

MySQL 中的 CAST 函数详解及常见用法

《MySQL中的CAST函数详解及常见用法》CAST函数是MySQL中用于数据类型转换的重要函数,它允许你将一个值从一种数据类型转换为另一种数据类型,本文给大家介绍MySQL中的CAST... 目录mysql 中的 CAST 函数详解一、基本语法二、支持的数据类型三、常见用法示例1. 字符串转数字2. 数字

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

MySQL count()聚合函数详解

《MySQLcount()聚合函数详解》MySQL中的COUNT()函数,它是SQL中最常用的聚合函数之一,用于计算表中符合特定条件的行数,本文给大家介绍MySQLcount()聚合函数,感兴趣的朋... 目录核心功能语法形式重要特性与行为如何选择使用哪种形式?总结深入剖析一下 mysql 中的 COUNT

MySQL 中 ROW_NUMBER() 函数最佳实践

《MySQL中ROW_NUMBER()函数最佳实践》MySQL中ROW_NUMBER()函数,作为窗口函数为每行分配唯一连续序号,区别于RANK()和DENSE_RANK(),特别适合分页、去重... 目录mysql 中 ROW_NUMBER() 函数详解一、基础语法二、核心特点三、典型应用场景1. 数据分

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN