15

如果程序集已在本地计算机上的 GAC(全局程序集缓存)中注册,以编程方式检查的最简单方法是什么?是否有一些易于使用的 .NET API,我可以将它提供给程序集 DLL 或程序集对象本身的位置,以检查它是否存在于本地计算机上的 GAC 中?在我的情况下,我正在检查的程序集将已经加载到程序检查的当前 AppDomain 中,所以我不确定调用Assembly.ReflectionOnlyLoad和捕获异常是否会像我在其他帖子中看到的那样工作,而且这看起来有点 hacky .

理想情况下,我想避免调用外部可执行文件gacutil.exe来检查。

4

2 回答 2

23

这个问题与以下问题非常相似,但我的问题更精确一些,而且两者都没有被接受的答案,而且所提供的答案似乎都不完整或最优:

最初我认为以下是最好的方法,但是除非您指定程序集的全名,否则它不起作用,并且由于 try/catch 有点骇人听闻,但它很简单并且适用于许多情况:

public static class GacUtil
{
    public static bool IsAssemblyInGAC(string assemblyFullName)
    {
        try
        {
            return Assembly.ReflectionOnlyLoad(assemblyFullName)
                           .GlobalAssemblyCache;
        }
        catch
        {
            return false;
        }
    }

    public static bool IsAssemblyInGAC(Assembly assembly)
    {
        return assembly.GlobalAssemblyCache;
    }
}

这是一种更好的方法,通过使用Fusion API无需 try/catch 即可工作。这是一堆更多的代码,但它适用于部分程序集名称:

public static class GacUtil
{
    [DllImport("fusion.dll")]
    private static extern IntPtr CreateAssemblyCache(
        out IAssemblyCache ppAsmCache, 
        int reserved);

    [ComImport]
    [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
    [Guid("e707dcde-d1cd-11d2-bab9-00c04f8eceae")]
    private interface IAssemblyCache
    {
        int Dummy1();

        [PreserveSig()]
        IntPtr QueryAssemblyInfo(
            int flags, 
            [MarshalAs(UnmanagedType.LPWStr)] string assemblyName, 
            ref AssemblyInfo assemblyInfo);

        int Dummy2();
        int Dummy3();
        int Dummy4();
    }

    [StructLayout(LayoutKind.Sequential)]
    private struct AssemblyInfo
    {
        public int cbAssemblyInfo;
        public int assemblyFlags;
        public long assemblySizeInKB;

        [MarshalAs(UnmanagedType.LPWStr)]
        public string currentAssemblyPath;

        public int cchBuf;
    }

    public static bool IsAssemblyInGAC(string assemblyName)
    {
        var assembyInfo = new AssemblyInfo { cchBuf = 512 };
        assembyInfo.currentAssemblyPath = new string('\0', assembyInfo.cchBuf);

        IAssemblyCache assemblyCache;

        var hr = CreateAssemblyCache(out assemblyCache, 0);

        if (hr == IntPtr.Zero)
        {
            hr = assemblyCache.QueryAssemblyInfo(
                1, 
                assemblyName, 
                ref assembyInfo);

            if (hr != IntPtr.Zero)
            {
                return false;
            }

            return true;
        }

        Marshal.ThrowExceptionForHR(hr.ToInt32());
        return false;
    }
}
于 2013-10-18T21:32:48.337 回答
-1

检查 CodeBase 是否为空

if (asm.CodeBase == null) {
     // IN GAC
}
于 2015-12-28T22:38:00.493 回答