web-dev-qa-db-ja.com

関数名を文字列として渡すことにより、任意の関数を動的に呼び出す

インスタンスを作成してその機能を動的に実行するプロセスを自動化するにはどうすればよいですか?

ありがとう

編集:パラメータを渡すオプションも必要です。ありがとう

25
Josh

インスタンスを作成するために、パラメーターなしのコンストラクターを呼び出しますか?タイプも文字列として指定されていますか、それともジェネリックメソッドにできますか?例えば:

// All error checking omitted. In particular, check the results
// of Type.GetType, and make sure you call it with a fully qualified
// type name, including the Assembly if it's not in mscorlib or
// the current Assembly. The method has to be a public instance
// method with no parameters. (Use BindingFlags with GetMethod
// to change this.)
public void Invoke(string typeName, string methodName)
{
    Type type = Type.GetType(typeName);
    object instance = Activator.CreateInstance(type);
    MethodInfo method = type.GetMethod(methodName);
    method.Invoke(instance, null);
}

または

public void Invoke<T>(string methodName) where T : new()
{
    T instance = new T();
    MethodInfo method = typeof(T).GetMethod(methodName);
    method.Invoke(instance, null);
}
48
Jon Skeet

コンストラクターを呼び出すには、 Activator.CreateInstance を使用します。それはあなたの人生を容易にするためにたくさんのオーバーロードを持っています。

コンストラクタが parameterless の場合:

object instance = Activator.CreateInstance(type)

パラメータ が必要な場合:

object instance =  Activator.CreateInstance(type, param1, param2)

メソッドを呼び出すには、 Type オブジェクトを取得したら、 GetMethod を呼び出して method を取得し、次に- Invoke (パラメーターありまたはなし)で呼び出します。必要な場合、Invokeは、呼び出している関数の戻り値(またはvoidメソッドの場合はnull)も提供します。

もう少し詳細なサンプルの場合(コンソールアプリに貼り付けて実行):

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.IO;
using System.Reflection;

namespace Test
{
    public static class Invoker
    {
        public static object CreateAndInvoke(string typeName, object[] constructorArgs, string methodName, object[] methodArgs)
        {
            Type type = Type.GetType(typeName);
            object instance = Activator.CreateInstance(type, constructorArgs);

            MethodInfo method = type.GetMethod(methodName);
            return method.Invoke(instance, methodArgs);
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            // Default constructor, void method
            Invoker.CreateAndInvoke("Test.Tester", null, "TestMethod", null);

            // Constructor that takes a parameter
            Invoker.CreateAndInvoke("Test.Tester", new[] { "constructorParam" }, "TestMethodUsingValueFromConstructorAndArgs", new object[] { "moo", false });

            // Constructor that takes a parameter, invokes a method with a return value
            string result = (string)Invoker.CreateAndInvoke("Test.Tester", new object[] { "constructorValue" }, "GetContstructorValue", null);
            Console.WriteLine("Expect [constructorValue], got:" + result);

            Console.ReadKey(true);
        }
    }

    public class Tester
    {
        public string _testField;

        public Tester()
        {
        }

        public Tester(string arg)
        {
            _testField = arg;
        }

        public void TestMethod()
        {
            Console.WriteLine("Called TestMethod");
        }

        public void TestMethodWithArg(string arg)
        {
            Console.WriteLine("Called TestMethodWithArg: " + arg);
        }

        public void TestMethodUsingValueFromConstructorAndArgs(string arg, bool arg2)
        {
            Console.WriteLine("Called TestMethodUsingValueFromConstructorAndArg " + arg + " " + arg2 + " " + _testField);
        }

        public string GetContstructorValue()
        {
            return _testField;
        }
    }
}
20
Nader Shirazie

呼び出すメソッドがパラメーターを取らないと仮定します。

public void InvokeMethod(Type type, string methodName)
{
    object instance = Activator.CreateInstance(type);
    MethodInfo method = type.GetMethod(methodName, BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);

    method.Invoke(instance, null);
}
7
Fredrik Mörk

あなたの問題はここではあまりにも一般的すぎると思います。ここでは、特定の前提を備えたソリューションを提供しています。

前提:typeName(文字列)、methodName(文字列)、およびパラメーター(SomeType)があります。

public static void InvokeMethod(string typeName, string methodName, SomeType objSomeType) {
      Type type = Type.GetType(typeName);
      if(type==null) {
        return;
      }
      object instance = Activator.CreateInstance(type); //Type must have a parameter-less contructor, or no contructor.   
      MethodInfo methodInfo =type.GetMethod(methodName, BindingFlags.Instance | BindingFlags.Public);
      if(methodInfo==null) {
        return;
      }
      methodInfo.Invoke(instance, new[] { objSomeType });  
    } 

私の仮定が間違っているかどうか知らせてください。

3

パラメーターを動的に渡すにはここでは、params string [] argsを使用しました。これは、関数によってパラメーターの数が異なるためです。

public void Invoke(string typeName,string functionName,params string[] args)
    {

     Type type = Type.GetType(typeName);
     dynamic c=Activator.CreateInstance(type);
     //args contains the parameters(only string type)
     type.InvokeMember(functionName,BindingFlags.InvokeMethod,null,c,args);   

    }
2
chirag pathak