有没有一种简单的方法来读取应用程序已经嵌入的清单文件?
我在考虑替代数据流的思路?
Windows 清单文件是 Win32 资源。换句话说,它们嵌入到 EXE 或 DLL 的末尾。您可以使用 LoadLibraryEx、FindResource、LoadResource 和 LockResource 来加载嵌入的资源。
这是一个提取自己的清单的简单示例...
BOOL CALLBACK EnumResourceNameCallback(HMODULE hModule, LPCTSTR lpType,
LPWSTR lpName, LONG_PTR lParam)
{
HRSRC hResInfo = FindResource(hModule, lpName, lpType);
DWORD cbResource = SizeofResource(hModule, hResInfo);
HGLOBAL hResData = LoadResource(hModule, hResInfo);
const BYTE *pResource = (const BYTE *)LockResource(hResData);
TCHAR filename[MAX_PATH];
if (IS_INTRESOURCE(lpName))
_stprintf_s(filename, _T("#%d.manifest"), lpName);
else
_stprintf_s(filename, _T("%s.manifest"), lpName);
FILE *f = _tfopen(filename, _T("wb"));
fwrite(pResource, cbResource, 1, f);
fclose(f);
UnlockResource(hResData);
FreeResource(hResData);
return TRUE; // Keep going
}
int _tmain(int argc, _TCHAR* argv[])
{
const TCHAR *pszFileName = argv[0];
HMODULE hModule = LoadLibraryEx(pszFileName, NULL, LOAD_LIBRARY_AS_DATAFILE);
EnumResourceNames(hModule, RT_MANIFEST, EnumResourceNameCallback, NULL);
FreeLibrary(hModule);
return 0;
}
或者,您可以使用 Windows SDK 中的 MT.EXE:
>mt -inputresource:dll_with_manifest.dll;#1 -out:extracted.manifest
您可以使用命令行清单工具提取/替换/合并/验证清单mt.exe
,该工具是 Windows SDK 的一部分:
C:\Program Files\Microsoft SDKs\Windows\v6.1>mt /?
Microsoft (R) Manifest Tool version 5.2.3790.2075
...
> To extract manifest out of a dll:
mt.exe -inputresource:dll_with_manifest.dll;#1 -out:extracted.manifest
不同地点:
在记事本中打开文件。事情是纯文本的。
这里有一个清单查看器工具——我不知道作者是否会提供源代码。
如果 Resource Tuner 支持 x64 代码,它会很好,但截至今天,它仍然只适用于 32 位应用程序。Resource Hacker(最新的公开测试版)确实支持 x86 和 x64,可从此处获得: http: //angusj.com/resourcehacker/
在编译的应用程序中查看/编辑清单的最简单方法是使用资源调谐器: http ://www.restuner.com/tour-manifest.htm
在某些情况下,它比 MS 的 mt.exe 更强大,而且它是一个可视化工具。
从 Roger 的代码开始工作,这是我使用的代码。它假定 Manifest 位于 id #1。我猜这是 .exe 的默认设置。请参阅 Wedge 的评论,如果您使用 DLL,您可能还必须检查 id #2。
HMODULE module = ::LoadLibraryEx(pathname, NULL, LOAD_LIBRARY_AS_DATAFILE);
if (module == NULL)
return false;
HRSRC resInfo = ::FindResource(module, MAKEINTRESOURCE(1), RT_MANIFEST); // resource id #1 should be the manifest
if (resInfo) {
HGLOBAL resData = ::LoadResource(module, resInfo);
DWORD resSize = ::SizeofResource(module, resInfo);
if (resData && resSize) {
const char *res = (const char *)::LockResource(resData); // the manifest
if (res) {
// got the manifest
}
::UnlockResource(resData);
}
::FreeResource(resData);
}
::FreeLibrary(module);
通过从项目中删除开发人员许可证 ( *_TemporaryKey.pfx
) 或更改 .pfx 的名称来解决此问题。
附带提醒:请记住,清单也可以是与应用程序同名的独立文件(由“.manifest”扩展)。
因此,如果您想检查在运行时真正使用了哪个清单,则必须考虑到这一点。