我可以获取当前未加载的程序集列表

时间:2012-05-30 04:58:28

标签: c# reflection

我希望允许使用我的产品的开发人员能够通过实现接口扩展产品,然后将程序集放入执行文件夹中。 我应该如何找到这些类型,我是否必须通过文件夹中的每个DLL运行,或者我可以避免那些属于原始应用程序的那些?

4 个答案:

答案 0 :(得分:1)

为什么不使用microsoft's solutions
据我了解,它解决了你正在寻找的东西

答案 1 :(得分:0)

您可以将新程序集放入不同的文件夹(例如调用或插件或扩展),然后在运行时实例化它们;或者,维护产品程序集的列表并使用Path.GetFiles并删除列表中显示的所有程序集。

答案 2 :(得分:0)

我将此用于用户定义的工具。

private static void getImplementedTypes(Type baseType, Assembly assembly, IList<Type> list) {
    Type[] types = assembly.GetExportedTypes();
    foreach (Type t in types) {
        if (baseType.IsInterface) {
            Type[] interfaces = t.GetInterfaces();
            foreach (Type i in interfaces) {
                if (i == baseType) list.Add(t);
            }
        }
        else {
            if ((!list.Contains(t)) && (t.IsSubclassOf(baseType)) && (!t.IsAbtract)) {
                list.Add(t);
            }
        }
    }
    return n;
}

在循环中,我遍历工具目录中Directory.GetFiles找到的所有DLL(或EXE):

Assembly assembly = Assembly.LoadFile("toolbox.dll");
List<Type> types = new List<Type>();
getImplementedTypes(typeof(ToolBase), assembly, types);
ToolBase theTool = Activator.CreateInstance(type, true) as ToolBase;

这适用于接口和基类。

我不知道以不同的方式查找已实现的类的方法。 这可能要花点时间。因此,如果你知道要搜索哪些DLL,只能循环遍历它们。

答案 3 :(得分:0)

最后,我选择简单地加载并搜索我在垃圾箱中找到的所有dll。 这是大部分代码。关键功能它是“IsAssignableToGenericType”函数,它找到我正在寻找的通用接口。

我相信这是提供大部分解决方案的链接 Implementations of interface through Reflection

static AssemblyLocator()
    {
        AllDlls = GetAllDlls();
        SubscribersInBin = GetSubscribersInBin();
    }

    public static IEnumerable<Type> TypesImplementingInterface(Assembly[] assemblies, Type desiredType)
    {
        return assemblies
            .SelectMany(assembly => assembly.GetTypes())
            .Where(type => IsAssignableToGenericType(type, desiredType));
    }

    public static bool IsAssignableToGenericType(Type givenType, Type genericType)
    {
        if (givenType == null) throw new ArgumentNullException("givenType");
        if (genericType == null) throw new ArgumentNullException("genericType");

        var interfaceTypes = givenType.GetInterfaces();

        foreach (var it in interfaceTypes)
        {
            if (it.IsGenericType)
            {
                if (it.GetGenericArguments()[0].Name.Equals(genericType.GetGenericArguments()[0].Name))
                    return true;
            }
        }

        Type baseType = givenType.BaseType;
        if (baseType == null) return false;

        return (baseType.IsGenericType &&
            baseType.GetGenericTypeDefinition() == genericType) ||
            IsAssignableToGenericType(baseType, genericType);
    }

    private static ReadOnlyCollection<string> GetAllDlls()
    {
        string binFolder = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location);
        IList<string> dllFiles = Directory.GetFiles(binFolder, "*.dll", SearchOption.TopDirectoryOnly).ToList();
        return new ReadOnlyCollection<string>(dllFiles);
    }

    private static ReadOnlyCollection<Type> GetSubscribersInBin()
    {
        IList<Assembly> assembliesFoundInBin = new List<Assembly>();
        foreach (var item in AllDlls)
        {
            var assembly = System.Reflection.Assembly.LoadFrom(item);
            assembliesFoundInBin.Add(assembly);
        }

        var typesInBin = TypesImplementingInterface(assembliesFoundInBin.ToArray(), typeof(ISubscriber<T>));
        return new ReadOnlyCollection<Type>(typesInBin.ToList<Type>());
    }