Создание объекта с типом, определенным во время выполнения - PullRequest
58 голосов
/ 11 июня 2009

Я нахожусь в ситуации, когда я хотел бы создать объект типа, который будет определен во время выполнения. Мне также нужно выполнить явное приведение к этому типу.

Примерно так:

static void castTest(myEnum val)
{
    //Call a native function that returns a pointer to a structure
    IntPtr = someNativeFunction(..params..);

    //determine the type of the structure based on the enum value
    Type structType = getTypeFromEnum(val);

    structType myStruct = (structType)Marshal.PtrToStructure(IntPtr, structType);
}

Это явно недействительный код, но я надеюсь, что он передает суть того, что я пытаюсь сделать. Метод, над которым я на самом деле работаю, должен выполнить маршалинг для ~ 35 различных типов. У меня есть несколько других методов, которые должны будут сделать что-то подобное с тем же набором типов. Итак, я хотел бы изолировать логику определения типа от этих методов, чтобы мне нужно было написать ее только один раз, чтобы методы оставались чистыми и читаемыми.

Я должен признать, что я новичок в дизайне. Кто-нибудь может предложить хороший подход к этой проблеме? Я подозреваю, что может быть подходящий шаблон проектирования, о котором я не знаю.

Ответы [ 6 ]

113 голосов
/ 11 июня 2009

Существует несколько способов создания объекта определенного типа на лету, один из них:

// determine type here
var type = typeof(MyClass);

// create an object of the type
var obj = (MyClass)Activator.CreateInstance(type);

И вы получите экземпляр MyClass в obj.

Другой способ - использовать отражение:

// get type information
var type = typeof(MyClass);

// get public constructors
var ctors = type.GetConstructors(BindingFlags.Public);

// invoke the first public constructor with no parameters.
var obj = ctors[0].Invoke(new object[] { });

И из одного возвращенного ConstructorInfo вы можете вызвать его с помощью аргументов Invoke () и получить экземпляр класса, как если бы вы использовали оператор «new».

14 голосов
/ 11 июня 2009

Вы можете в основном делать то, что вы описываете, но так как вы не знаете тип во время компиляции, вам придется держать экземпляр свободно набранным; проверяйте его тип в каждой точке, в которой вы его используете, и затем преобразуйте его соответствующим образом (в C # 4.0, который поддерживает dynamic ) это не требуется:

Type type = CustomGetTypeMethod();
var obj = Activator.CreateInstance(type);

...


if(obj is MyCustomType)
{
    ((MyCustomType)obj).Property1;
}
else if (obj is MyOtherCustomType)
{
    ((MyOtherCustomType)obj).Property2;
}
10 голосов
/ 11 июня 2009

Я думаю, что вы ищете Activator.CreateInstance

6 голосов
/ 11 июня 2009

Создать экземпляр среды выполнения, определенной Type, легко, используя Activator.CreateInstance, как уже упоминали другие. Однако приведение к нему, как вы делаете в своем примере в строке Marshal.PtrToStructure, невозможно, так как тип должен быть известен во время компиляции для преобразования. Также обратите внимание, что Activator.CreateInstance нельзя использовать вместе с IntPtr.

Если у ваших типов есть общий базовый класс (отличный от Object), вы можете привести его к указанному базовому типу и вызвать его функции. В противном случае вызов функций будет возможен только с помощью отражения.

Так что либо:

static void castTest(myEnum val)
{
  //Call a native function that returns a pointer to a structure
  IntPtr val = someNativeFunction(..params..);

  //determine the type of the structure based on the enum value
  Type structType = getTypeFromEnum(val);

  BaseClass myStruct = (BaseClass)Marshal.PtrToStructure(IntPtr, structType);
  myStruct.SomeFunctionDeclaredInBaseClass();
}

Или:

static void castTest(myEnum val)
{
  //Call a native function that returns a pointer to a structure
  IntPtr val = someNativeFunction(..params..);

  //determine the type of the structure based on the enum value
  Type structType = getTypeFromEnum(val);

  object myStruct = Marshal.PtrToStructure(IntPtr, structType);
  MemberInfo[] function = FindMembers(MemberTypes.Method, BindingFlags.Public | BindingFlags.Instance,
    (MemberFilter)delegate(MemberInfo info, object filter)
    {
      return info.Name == filter.ToString();
    }, "SomeFunction");
  if (mi.Length > 0 && mi[0] is MethodInfo)
    ((MethodInfo)mi[0]).Invoke(myStruct, ..params..);
}
1 голос
/ 06 ноября 2012

Вы можете пойти динамически:

using System;

namespace TypeCaster
{
    class Program
    {
        internal static void Main(string[] args)
        {
            Parent p = new Parent() { name = "I am the parent", type = "TypeCaster.ChildA" };
            dynamic a = Convert.ChangeType(new ChildA(p.name), Type.GetType(p.type));
            Console.WriteLine(a.Name);

            p.type = "TypeCaster.ChildB";
            dynamic b = Convert.ChangeType(new ChildB(p.name), Type.GetType(p.type));
            Console.WriteLine(b.Name);
        }
    }

    internal class Parent
    {
        internal string type { get; set; }
        internal string name { get; set; }

        internal Parent() { }
    }

    internal class ChildA : Parent
    {
        internal ChildA(string name)
        {
            base.name = name + " in A";
        }

        public string Name
        {
            get { return base.name; }
        }
    }

    internal class ChildB : Parent
    {
        internal ChildB(string name)
        {
            base.name = name + " in B";
        }

        public string Name
        {
            get { return base.name; }
        }
    }
}
0 голосов
/ 11 декабря 2015
 methodName = NwSheet.Cells[rCnt1, cCnt1 - 2].Value2;
                            Type nameSpace=typeof(ReadExcel);
                            Type metdType = Type.GetType(nameSpace.Namespace + "." + methodName);
                            //ConstructorInfo magicConstructor = metdType.GetConstructor(Type.EmptyTypes);
                            //object magicClassObject = magicConstructor.Invoke(new object[] { });
                            object magicClassObject = Activator.CreateInstance(metdType);
                            MethodInfo mthInfo = metdType.GetMethod("fn_"+methodName);
                            StaticVariable.dtReadData.Clear();
                            for (iCnt = cCnt1 + 4; iCnt <= ShtRange.Columns.Count; iCnt++)
                            {
                                temp = NwSheet.Cells[1, iCnt].Value2;
                                StaticVariable.dtReadData.Add(temp.Trim(), Convert.ToString(NwSheet.Cells[rCnt1, iCnt].Value2));
                            }


                            //if (Convert.ToString(NwSheet.Cells[rCnt1, cCnt1 - 2].Value2) == "fn_AddNum" || Convert.ToString(NwSheet.Cells[rCnt1, cCnt1 - 2].Value2) == "fn_SubNum")
                            //{
                            //    //StaticVariable.intParam1 = Convert.ToInt32(NwSheet.Cells[rCnt1, cCnt1 + 4].Value2);
                            //    //StaticVariable.intParam2 = Convert.ToInt32(NwSheet.Cells[rCnt1, cCnt1 + 5].Value2);
                            //    object[] mParam1 = new object[] { Convert.ToInt32(StaticVariable.dtReadData["InParam1"]), Convert.ToInt32(StaticVariable.dtReadData["InParam2"]) };
                            //    object result = mthInfo.Invoke(this, mParam1);
                            //    StaticVariable.intOutParam1 = Convert.ToInt32(result);
                            //    NwSheet.Cells[rCnt1, cCnt1 + 2].Value2 = Convert.ToString(StaticVariable.intOutParam1) != "" ? Convert.ToString(StaticVariable.intOutParam1) : String.Empty;
                            //}

                            //else
                            //{
                                object[] mParam = new object[] { };
                                mthInfo.Invoke(magicClassObject, mParam);
...