TypeBuilder.GetConstructor(Type, ConstructorInfo) Método

Definición

Devuelve el constructor del tipo genérico construido especificado que corresponde al constructor especificado de la definición de tipo genérico.

public static System.Reflection.ConstructorInfo GetConstructor (Type type, System.Reflection.ConstructorInfo constructor);

Parámetros

type
Type

Tipo genérico construido cuyo constructor se devuelve.

constructor
ConstructorInfo

Constructor de la definición de tipo genérico de type, que especifica qué constructor de type devolver.

Devoluciones

Objeto ConstructorInfo que representa el constructor de type correspondiente a constructor, que especifica un constructor perteneciente a la definición de tipo genérico de type.

Excepciones

type no representa un tipo genérico.

o bien

type no es del tipo TypeBuilder.

o bien

El tipo declarativo de constructor no es una definición de tipo genérico.

o bien

El tipo declarativo de constructor no es una definición de tipo genérico de type.

Ejemplos

El ejemplo de código siguiente contiene código fuente para una clase genérica denominada Sample que tiene un parámetro de tipo denominado T. La clase tiene un campo denominado Field, de tipo Ty un método genérico denominado GM con su propio parámetro de tipo, denominado U. El método GM crea una instancia de Sample, sustituyendo su propio parámetro U de tipo para el parámetro type de Sampley almacena su parámetro de entrada en Field. Este código fuente se compila pero no se usa; Puede verlo con el Ildasm.exe (Desensamblador de IL) y compararlo con el código emitido por la clase Example.

El código de la clase Example muestra el uso del GetConstructor método para emitir código genérico. El Main método de clase Example crea un ensamblado dinámico que contiene una clase denominada Sample y usa el DefineGenericParameters método para que sea genérico agregando un parámetro de tipo denominado T. Un constructor sin parámetros y un campo denominado Field, de tipo T, se agregan a la clase Sample. Se agrega un método GM y se convierte en un método genérico mediante el MethodBuilder.DefineGenericParameters método . El parámetro type de GM se denomina U. Una vez definido el parámetro de tipo, la firma de GM se agrega mediante el MethodBuilder.SetSignature método . No hay ningún tipo de valor devuelto y ningún modificador obligatorio o personalizado, por lo que todos los parámetros de este método son null excepto parameterTypes; parameterTypes establece el tipo del único parámetro del método en U, el parámetro de tipo genérico del método. El cuerpo del método crea una instancia del tipo Sample<U> construido (Sample(Of U) en Visual Basic), asigna el parámetro del método a y, a Fieldcontinuación, imprime el valor de Field. El GetConstructor método se usa para crear un ConstructorInfo objeto que representa el constructor sin parámetros del tipo Sample<U> genérico construido en la OpCodes.Newobj instrucción que crea la instancia.

Se define un tipo ficticio para contener el método Mainde punto de entrada . En el cuerpo de Main, el método estático GM se invoca en el tipo Sample<int> genérico construido (Sample(Of Integer) en Visual Basic), con el tipo String sustituido por U.

Cuando se ejecuta el ejemplo de código, guarda el ensamblado emitido como TypeBuilderGetFieldExample.exe. Puede ejecutar TypeBuilderGetFieldExample.exe y puede usar el Ildasm.exe (Desensamblador de IL) para comparar el código emitido con el código de la Sample clase que se compila en el propio ejemplo de código.

using System;
using System.Reflection;
using System.Reflection.Emit;

// Compare the MSIL in this class to the MSIL
// generated by the Reflection.Emit code in class
// Example.
public class Sample<T>
{
  public T Field;
  public static void GM<U>(U val)
  {
    Sample<U> s = new Sample<U>();
    s.Field = val;
    Console.WriteLine(s.Field);
  }
}

public class Example
{
    public static void Main()
    {
        AppDomain myDomain = AppDomain.CurrentDomain;
        AssemblyName myAsmName =
            new AssemblyName("TypeBuilderGetFieldExample");
        AssemblyBuilder myAssembly = myDomain.DefineDynamicAssembly(
            myAsmName, AssemblyBuilderAccess.Save);
        ModuleBuilder myModule = myAssembly.DefineDynamicModule(
            myAsmName.Name,
            myAsmName.Name + ".exe");

        // Define the sample type.
        TypeBuilder myType = myModule.DefineType("Sample",
            TypeAttributes.Class | TypeAttributes.Public);

        // Add a type parameter, making the type generic.
        string[] typeParamNames = {"T"};
        GenericTypeParameterBuilder[] typeParams =
            myType.DefineGenericParameters(typeParamNames);

        // Define a default constructor. Normally it would
        // not be necessary to define the default constructor,
        // but in this case it is needed for the call to
        // TypeBuilder.GetConstructor, which gets the default
        // constructor for the generic type constructed from
        // Sample<T>, in the generic method GM<U>.
        ConstructorBuilder ctor = myType.DefineDefaultConstructor(
            MethodAttributes.PrivateScope | MethodAttributes.Public |
            MethodAttributes.HideBySig | MethodAttributes.SpecialName |
            MethodAttributes.RTSpecialName);

        // Add a field of type T, with the name Field.
        FieldBuilder myField = myType.DefineField("Field",
            typeParams[0],
            FieldAttributes.Public);

        // Add a method and make it generic, with a type
        // parameter named U. Note how similar this is to
        // the way Sample is turned into a generic type. The
        // method has no signature, because the type of its
        // only parameter is U, which is not yet defined.
        MethodBuilder genMethod = myType.DefineMethod("GM",
            MethodAttributes.Public | MethodAttributes.Static);
        string[] methodParamNames = {"U"};
        GenericTypeParameterBuilder[] methodParams =
            genMethod.DefineGenericParameters(methodParamNames);

        // Now add a signature for genMethod, specifying U
        // as the type of the parameter. There is no return value
        // and no custom modifiers.
        genMethod.SetSignature(null, null, null,
            new Type[] { methodParams[0] }, null, null);

        // Emit a method body for the generic method.
        ILGenerator ilg = genMethod.GetILGenerator();
        // Construct the type Sample<U> using MakeGenericType.
        Type SampleOfU = myType.MakeGenericType( methodParams[0] );
        // Create a local variable to store the instance of
        // Sample<U>.
        ilg.DeclareLocal(SampleOfU);
        // Call the default constructor. Note that it is
        // necessary to have the default constructor for the
        // constructed generic type Sample<U>; use the
        // TypeBuilder.GetConstructor method to obtain this
        // constructor.
        ConstructorInfo ctorOfU = TypeBuilder.GetConstructor(
            SampleOfU, ctor);
        ilg.Emit(OpCodes.Newobj, ctorOfU);
        // Store the instance in the local variable; load it
        // again, and load the parameter of genMethod.
        ilg.Emit(OpCodes.Stloc_0);
        ilg.Emit(OpCodes.Ldloc_0);
        ilg.Emit(OpCodes.Ldarg_0);
        // In order to store the value in the field of the
        // instance of Sample<U>, it is necessary to have
        // a FieldInfo representing the field of the
        // constructed type. Use TypeBuilder.GetField to
        // obtain this FieldInfo.
        FieldInfo FieldOfU = TypeBuilder.GetField(
            SampleOfU, myField);
        // Store the value in the field.
        ilg.Emit(OpCodes.Stfld, FieldOfU);
        // Load the instance, load the field value, box it
        // (specifying the type of the type parameter, U), and
        // print it.
        ilg.Emit(OpCodes.Ldloc_0);
        ilg.Emit(OpCodes.Ldfld, FieldOfU);
        ilg.Emit(OpCodes.Box, methodParams[0]);
        MethodInfo writeLineObj =
            typeof(Console).GetMethod("WriteLine",
                new Type[] { typeof(object) });
        ilg.EmitCall(OpCodes.Call, writeLineObj, null);
        ilg.Emit(OpCodes.Ret);

        // Emit an entry point method; this must be in a
        // non-generic type.
        TypeBuilder dummy = myModule.DefineType("Dummy",
            TypeAttributes.Class | TypeAttributes.NotPublic);
        MethodBuilder entryPoint = dummy.DefineMethod("Main",
            MethodAttributes.Public | MethodAttributes.Static,
            null, null);
        ilg = entryPoint.GetILGenerator();
        // In order to call the static generic method GM, it is
        // necessary to create a constructed type from the
        // generic type definition for Sample. This can be any
        // constructed type; in this case Sample<int> is used.
        Type SampleOfInt =
            myType.MakeGenericType( typeof(int) );
        // Next get a MethodInfo representing the static generic
        // method GM on type Sample<int>.
        MethodInfo SampleOfIntGM = TypeBuilder.GetMethod(SampleOfInt,
            genMethod);
        // Next get a MethodInfo for GM<string>, which is the
        // instantiation of GM that Main calls.
        MethodInfo GMOfString =
            SampleOfIntGM.MakeGenericMethod( typeof(string) );
        // Finally, emit the call. Push a string onto
        // the stack, as the argument for the generic method.
        ilg.Emit(OpCodes.Ldstr, "Hello, world!");
        ilg.EmitCall(OpCodes.Call, GMOfString, null);
        ilg.Emit(OpCodes.Ret);

        myType.CreateType();
        dummy.CreateType();
        myAssembly.SetEntryPoint(entryPoint);
        myAssembly.Save(myAsmName.Name + ".exe");

        Console.WriteLine(myAsmName.Name + ".exe has been saved.");
    }
}

Comentarios

El GetConstructor método proporciona una manera de obtener un ConstructorInfo objeto que representa un constructor de un tipo genérico construido cuya definición de tipo genérico está representada por un TypeBuilder objeto .

Por ejemplo, supongamos que tiene un TypeBuilder objeto que representa el tipo G<T> en la sintaxis de C# (G(Of T) en Visual Basic, generic <T> ref class G en C++) y un ConstructorBuilder objeto que representa un constructor de G<T>. Supongamos que G<T> tiene un método genérico con un parámetro U de tipo que crea una instancia del tipo G<U>construido. Para emitir el código para crear una instancia del tipo construido, necesita un ConstructorInfo objeto que represente el constructor de este tipo construido; es decir, que cree una instancia de G<U>. Para ello, llame primero al MakeGenericType método en el TypeBuilder objeto , especificando el GenericTypeParameterBuilder objeto que representa U como argumento de tipo. A continuación, llame al GetConstructor método con el valor devuelto del MakeGenericType método como parámetro type y el ConstructorBuilder objeto que representa el constructor de G<U> como parámetro constructor. El valor devuelto es el ConstructorInfo objeto que necesita para emitir la llamada de función. En el ejemplo de código se muestra este escenario.

Se aplica a

Producto Versiones
.NET Core 1.0, Core 1.1, Core 2.0, Core 2.1, Core 2.2, Core 3.0, Core 3.1, 5, 6, 7, 8, 9
.NET Framework 2.0, 3.0, 3.5, 4.0, 4.5, 4.5.1, 4.5.2, 4.6, 4.6.1, 4.6.2, 4.7, 4.7.1, 4.7.2, 4.8, 4.8.1
.NET Standard 2.0 (package-provided), 2.1