6

我将动态获取dll。我需要加载 dll 并获取命名空间、类名来调用方法(方法名是静态的,它总是“OnStart()”)。基本上我需要通过加载dll来运行一个方法。有人可以帮忙吗!!!

4

4 回答 4

14

要加载程序集,您可以这样做:

Assembly assembly = Assembly.LoadFile(@"test.dll");

这假设您将磁盘上的程序集作为文件。如果您不这样做,例如从数据库中以字节数组的形式获取它们,则Assembly上还有其他方法可以帮助您在加载后为您提供一个 Assembly 对象。

要遍历程序集中的所有类,您可以这样做:

Assembly assembly = Assembly.LoadFile(@"test.dll");
foreach (Type type in assembly.GetTypes())
{
    if (type.IsClass)
    {
        ...
    }
}

要查找 OnStart 静态方法,您可以执行以下操作:

Assembly assembly = Assembly.LoadFile(@"test.dll");
foreach (Type type in assembly.GetTypes())
{
    if (type.IsClass)
    {
        MethodInfo method = type.GetMethod("OnStart",
            BindingFlags.Static | BindingFlags.Public);
        if (method != null)
        {
            ...
        }
    }
}

要调用该方法,您可以这样做:

Assembly assembly = Assembly.LoadFile(@"test.dll");
foreach (Type type in assembly.GetTypes())
{
    if (type.IsClass)
    {
        MethodInfo method = type.GetMethod("OnStart",
            BindingFlags.Static | BindingFlags.Public);
        if (method != null)
        {
            method.Invoke(null, new Object[0]); // assumes no parameters
            break; // no need to look for more methods, unless you got multiple?
        }
    }
}

如果需要将参数传递给方法,可以将它们放在对象数组中:

Object[] arguments = new Object[] { arg1, arg2, arg3 ... };
method.Invoke(null, arguments);

上面的代码可以通过使用Linq为我们找到方法折叠成如下:

Assembly assembly = Assembly.LoadFile(@"test.dll");
var method = (from type in assembly.GetTypes()
              where type.IsClass
              let onStartMethod = type.GetMethod("OnStart",
                  BindingFlags.Static | BindingFlags.Public)
              where onStartMethod != null
              select onStartMethod).FirstOrDefault();
if (method != null)
{
    method.Invoke(null, new Object[0]); // assumes no parameters
}
于 2009-10-20T09:49:47.147 回答
1
       object result = null;

        using (StreamReader reader = new StreamReader(ASSEMBLYPATH, Encoding.GetEncoding(1252), false))
        {
            byte[] b = new byte[reader.BaseStream.Length];
            reader.BaseStream.Read(b, 0, Convert.ToInt32(reader.BaseStream.Length));
            reader.Close();

            Assembly asm = AppDomain.CurrentDomain.Load(b);
            Type typeClass = asm.GetType(CLASSFULLNAME); // including namespace
            MethodInfo mi = typeClass.GetMethod("OnStart");
            ConstructorInfo ci = typeClass.GetConstructor(Type.EmptyTypes);
            object responder = ci.Invoke(null);

            // set parameters
            object[] parameters = new object[1];
            parameters[0] = null;  // no params


            result = mi.Invoke(responder, parameters);
        }

使用此代码的优点是程序集在使用后会被卸载,因此您可以在调用该方法后安全地删除 dll。

于 2009-10-20T09:49:29.750 回答
0

在这里查看:http: //dotnetguts.blogspot.com/2008/12/reflection-in-c-list-of-class-name.html

在这里调用一个方法:http ://www.csharphelp.com/archives/archive200.html

如果您在这些链接中搜索更多术语,您会发现更多信息。

于 2009-10-20T09:44:01.783 回答
0

在运行时命名空间只是成为类型名称的一部分。

所以你需要:

  1. 加载程序集
  2. 获取所需类型的 Type 实例。
  3. 获取MethodInfo要调用的方法。
  4. 调用方法。

其中 2-4 个很容易。1可能容易,也可能不容易,这取决于装配的位置。假设可以通过正常的装配负载(“探测”)找到装配。这将调用一个不带参数但有返回值的类型的公共静态方法。

var asm = Assembly.Load(assemblyName);
var t = asm.GetType(typeName);
// Pass array of parameter types to resolve between overloads (here no arguments).
var m = t.GetMathod(methodName, BindingFlags.Static, null, new Type[] {}, null);
// Pass no "this" or arguments.
var res = (resultType) m.Invoke(null, null);

这里的许多细节将取决于您要调用的程序集、类型和方法的细节。

于 2009-10-20T09:49:12.793 回答