Прочитать на английском

Поделиться через


MethodBase.Invoke Метод

Определение

Вызывает метод или конструктор, отражаемый этим экземпляром MethodInfo.

Перегрузки

Invoke(Object, Object[])

Вызывает метод или конструктор, представленный текущим экземпляром, используя указанные параметры.

Invoke(Object, BindingFlags, Binder, Object[], CultureInfo)

При переопределении в производном классе вызывает отражаемый метод или конструктор с заданными параметрами.

Invoke(Object, Object[])

Исходный код:
MethodBase.cs
Исходный код:
MethodBase.cs
Исходный код:
MethodBase.cs

Вызывает метод или конструктор, представленный текущим экземпляром, используя указанные параметры.

public virtual object Invoke(object obj, object[] parameters);
public object? Invoke(object? obj, object?[]? parameters);
public object Invoke(object obj, object[] parameters);

Параметры

obj
Object

Объект, для которого нужно вызвать метод или конструктор. Если метод является статическим, этот аргумент игнорируется. Если конструктор является статическим, этот аргумент должен иметь значение null или представлять экземпляр класса, который определяет конструктор.

parameters
Object[]

Список аргументов для вызываемого метода или конструктора. Это массив объектов, количество, порядок и тип которых должны соответствовать списку параметров вызываемого метода или конструктора. Если параметров нет, для parameters должно быть указано значение null.

Если метод или конструктор, представленный этим экземпляром, принимает параметр ref (ByRef в Visual Basic), не требуются никакие специальные атрибуты для вызова этого метода или конструктора с использованием этой функции. Любой объект этого массива, которому не присвоено значение явным образом, будет содержать значение по умолчанию для своего типа объекта. Для элементов ссылочного типа это значение равно null. Для элементов типа значения значение по умолчанию равно 0, 0,0 или false, в зависимости от типа конкретного элемента.

Возвращаемое значение

Объект, который содержит возвращаемое значение вызываемого метода, или null при вызове конструктора.

Реализации

Исключения

Параметр obj имеет значение null и метод не является статическим.

-или-

Этот метод не объявлен и не унаследован в классе obj.

-или-

Вызывается статический конструктор, а obj не имеет значения null и не является экземпляром класса, в котором объявлен этот конструктор.

Примечание. Вместо этого в .NET для приложений Магазина Windows или переносимой библиотеке классов выполните перехват Exception .

Элементы массива parameters не соответствуют подписи метода или конструктора, отраженного этим экземпляром.

Вызванный метод или конструктор создает исключение.

-или-

Текущий экземпляр представляет DynamicMethod, который содержит непроверяемый код. См. подраздел "Проверка" в разделе примечаний для DynamicMethod.

Массив parameters содержит неправильное число аргументов.

Вызывающий объект не имеет разрешение на выполнение метода или конструктора, представленного текущим экземпляром.

Примечание. Вместо этого в .NET для приложений Магазина Windows или переносимой библиотеке классов перехватите исключение MemberAccessExceptionбазового класса , .

Тип, объявляющий метод, является открытым универсальным типом. То есть свойство ContainsGenericParameters для объявляющего типа возвращает значение true.

Текущий экземпляр представляет MethodBuilder.

Примеры

В следующем примере кода демонстрируется поиск динамического метода с использованием отражения. Обратите внимание, что нельзя использовать MethodInfo объект из базового класса для вызова переопределенного метода в производном классе, так как поздняя привязка не может разрешить переопределения.

using System;
using System.Reflection;

public class MagicClass
{
    private int magicBaseValue;

    public MagicClass()
    {
        magicBaseValue = 9;
    }

    public int ItsMagic(int preMagic)
    {
        return preMagic * magicBaseValue;
    }
}

public class TestMethodInfo
{
    public static void Main()
    {
        // Get the constructor and create an instance of MagicClass

        Type magicType = Type.GetType("MagicClass");
        ConstructorInfo magicConstructor = magicType.GetConstructor(Type.EmptyTypes);
        object magicClassObject = magicConstructor.Invoke(new object[]{});

        // Get the ItsMagic method and invoke with a parameter value of 100

        MethodInfo magicMethod = magicType.GetMethod("ItsMagic");
        object magicValue = magicMethod.Invoke(magicClassObject, new object[]{100});

        Console.WriteLine("MethodInfo.Invoke() Example\n");
        Console.WriteLine("MagicClass.ItsMagic() returned: {0}", magicValue);
    }
}

// The example program gives the following output:
//
// MethodInfo.Invoke() Example
//
// MagicClass.ItsMagic() returned: 900

Комментарии

Это удобный метод, который вызывает перегрузку Invoke(Object, BindingFlags, Binder, Object[], CultureInfo) метода, передавая Default для invokeAttr и null для binder и culture.

Если вызываемый метод создает исключение, Exception.GetBaseException метод возвращает исходное исключение.

Чтобы вызвать статический метод с помощью его MethodInfo объекта, передайте null для obj.

Примечание

Если эта перегрузка метода используется для вызова конструктора экземпляра, объект, предоставленный для obj , повторно инициализируется, то есть выполняются все инициализаторы экземпляров. Возвращается значение null. Если вызывается конструктор класса, класс повторно инициализируется; то есть выполняются все инициализаторы классов. Возвращается значение null.

Примечание

Начиная с .NET Framework 2.0, этот метод можно использовать для доступа к не открытым членам, если вызывающему объекту предоставлен ReflectionPermissionReflectionPermissionFlag.RestrictedMemberAccess флаг, и если набор разрешений для не являющихся открытыми членами ограничен набором разрешений вызывающего объекта или его подмножеством. (См. раздел Вопросы безопасности для отражения.) Чтобы использовать эту функцию, приложение должно быть предназначено для .NET Framework 3.5 или более поздней версии.

Если параметр отраженного метода является типом значения, а соответствующий аргумент в parametersnull, среда выполнения передает инициализированный с нуля экземпляр типа значения.

См. также раздел

Применяется к

.NET 10 и другие версии
Продукт Версии
.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, 10
.NET Framework 1.1, 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 1.0, 1.1, 1.2, 1.3, 1.4, 1.5, 1.6, 2.0, 2.1
UWP 10.0

Invoke(Object, BindingFlags, Binder, Object[], CultureInfo)

Исходный код:
MethodBase.cs
Исходный код:
MethodBase.cs
Исходный код:
MethodBase.cs

При переопределении в производном классе вызывает отражаемый метод или конструктор с заданными параметрами.

public abstract object? Invoke(object? obj, System.Reflection.BindingFlags invokeAttr, System.Reflection.Binder? binder, object?[]? parameters, System.Globalization.CultureInfo? culture);
public abstract object Invoke(object obj, System.Reflection.BindingFlags invokeAttr, System.Reflection.Binder binder, object[] parameters, System.Globalization.CultureInfo culture);

Параметры

obj
Object

Объект, для которого нужно вызвать метод или конструктор. Если метод является статическим, этот аргумент игнорируется. Если конструктор является статическим, этот аргумент должен иметь значение null или представлять экземпляр класса, который определяет конструктор.

invokeAttr
BindingFlags

Битовая маска, которая содержит от нуля и больше битовых флагов из атрибута BindingFlags в разных сочетаниях.

binder
Binder

Объект, позволяющий осуществлять привязку, приведение типов аргументов, вызов элементов, а также поиск объектов MemberInfo с помощью отражения. Если значение параметра binder равно null, используется связыватель по умолчанию.

parameters
Object[]

Список аргументов для вызываемого метода или конструктора. Это массив объектов, количество, порядок и тип которых должны соответствовать списку параметров вызываемого метода или конструктора. При отсутствии параметров — значение null.

Если метод или конструктор, представленный этим экземпляром, принимает параметр ByRef, для вызова метода или конструктора с использованием этой функции специальные атрибуты не требуются. Любой объект этого массива, которому не присвоено значение явным образом, будет содержать значение по умолчанию для своего типа объекта. Для элементов ссылочного типа это значение равно null. Для элементов, хранящих значения, это значение равно 0, 0,0 или false (в зависимости от типа конкретного элемента).

culture
CultureInfo

Экземпляр объекта CultureInfo , используемого для управления приведением типов. Если значение этого объекта — null, для текущего потока используется CultureInfo. (Например, необходимо преобразовывать строку, которая представляет 1000, в значение Double, поскольку при разных языках и региональных параметрах 1000 представляется по-разному.)

Возвращаемое значение

Объект Object, содержащий возвращаемое значение вызванного метода, значение null для конструктора или значение null, если метод возвращает значение типа void. Перед вызовом метода или конструктора функция Invoke проверяет наличие у пользователя права доступа и допустимость параметров.

Реализации

Исключения

Параметр obj имеет значение null и метод не является статическим.

-или-

Этот метод не объявлен и не унаследован в классе obj.

-или-

Вызывается статический конструктор, а obj не имеет значения null и не является экземпляром класса, в котором объявлен этот конструктор.

Тип параметра parameters не соответствует подписи метода или конструктора, отраженного этим экземпляром.

Массив parameters содержит неправильное число аргументов.

Вызванный метод или конструктор создает исключение.

Вызывающий объект не имеет разрешение на выполнение метода или конструктора, представленного текущим экземпляром.

Тип, объявляющий метод, является открытым универсальным типом. То есть свойство ContainsGenericParameters для объявляющего типа возвращает значение true.

Примеры

В следующем примере показаны все члены класса с System.Reflection.Binder использованием перегрузки Type.InvokeMember. Частный метод CanConvertFrom находит совместимые типы для данного типа. Еще один пример вызова членов в сценарии пользовательской привязки см. в разделе Динамическая загрузка и использование типов.

using System;
using System.Reflection;
using System.Globalization;

public class MyBinder : Binder
{
    public MyBinder() : base()
    {
    }
    private class BinderState
    {
        public object[] args;
    }
    public override FieldInfo BindToField(
        BindingFlags bindingAttr,
        FieldInfo[] match,
        object value,
        CultureInfo culture
        )
    {
        if(match == null)
            throw new ArgumentNullException("match");
        // Get a field for which the value parameter can be converted to the specified field type.
        for(int i = 0; i < match.Length; i++)
            if(ChangeType(value, match[i].FieldType, culture) != null)
                return match[i];
        return null;
    }
    public override MethodBase BindToMethod(
        BindingFlags bindingAttr,
        MethodBase[] match,
        ref object[] args,
        ParameterModifier[] modifiers,
        CultureInfo culture,
        string[] names,
        out object state
        )
    {
        // Store the arguments to the method in a state object.
        BinderState myBinderState = new BinderState();
        object[] arguments = new Object[args.Length];
        args.CopyTo(arguments, 0);
        myBinderState.args = arguments;
        state = myBinderState;
        if(match == null)
            throw new ArgumentNullException();
        // Find a method that has the same parameters as those of the args parameter.
        for(int i = 0; i < match.Length; i++)
        {
            // Count the number of parameters that match.
            int count = 0;
            ParameterInfo[] parameters = match[i].GetParameters();
            // Go on to the next method if the number of parameters do not match.
            if(args.Length != parameters.Length)
                continue;
            // Match each of the parameters that the user expects the method to have.
            for(int j = 0; j < args.Length; j++)
            {
                // If the names parameter is not null, then reorder args.
                if(names != null)
                {
                    if(names.Length != args.Length)
                        throw new ArgumentException("names and args must have the same number of elements.");
                    for(int k = 0; k < names.Length; k++)
                        if(String.Compare(parameters[j].Name, names[k].ToString()) == 0)
                            args[j] = myBinderState.args[k];
                }
                // Determine whether the types specified by the user can be converted to the parameter type.
                if(ChangeType(args[j], parameters[j].ParameterType, culture) != null)
                    count += 1;
                else
                    break;
            }
            // Determine whether the method has been found.
            if(count == args.Length)
                return match[i];
        }
        return null;
    }
    public override object ChangeType(
        object value,
        Type myChangeType,
        CultureInfo culture
        )
    {
        // Determine whether the value parameter can be converted to a value of type myType.
        if(CanConvertFrom(value.GetType(), myChangeType))
            // Return the converted object.
            return Convert.ChangeType(value, myChangeType);
        else
            // Return null.
            return null;
    }
    public override void ReorderArgumentArray(
        ref object[] args,
        object state
        )
    {
        // Return the args that had been reordered by BindToMethod.
        ((BinderState)state).args.CopyTo(args, 0);
    }
    public override MethodBase SelectMethod(
        BindingFlags bindingAttr,
        MethodBase[] match,
        Type[] types,
        ParameterModifier[] modifiers
        )
    {
        if(match == null)
            throw new ArgumentNullException("match");
        for(int i = 0; i < match.Length; i++)
        {
            // Count the number of parameters that match.
            int count = 0;
            ParameterInfo[] parameters = match[i].GetParameters();
            // Go on to the next method if the number of parameters do not match.
            if(types.Length != parameters.Length)
                continue;
            // Match each of the parameters that the user expects the method to have.
            for(int j = 0; j < types.Length; j++)
                // Determine whether the types specified by the user can be converted to parameter type.
                if(CanConvertFrom(types[j], parameters[j].ParameterType))
                    count += 1;
                else
                    break;
            // Determine whether the method has been found.
            if(count == types.Length)
                return match[i];
        }
        return null;
    }
    public override PropertyInfo SelectProperty(
        BindingFlags bindingAttr,
        PropertyInfo[] match,
        Type returnType,
        Type[] indexes,
        ParameterModifier[] modifiers
        )
    {
        if(match == null)
            throw new ArgumentNullException("match");
        for(int i = 0; i < match.Length; i++)
        {
            // Count the number of indexes that match.
            int count = 0;
            ParameterInfo[] parameters = match[i].GetIndexParameters();
            // Go on to the next property if the number of indexes do not match.
            if(indexes.Length != parameters.Length)
                continue;
            // Match each of the indexes that the user expects the property to have.
            for(int j = 0; j < indexes.Length; j++)
                // Determine whether the types specified by the user can be converted to index type.
                if(CanConvertFrom(indexes[j], parameters[j].ParameterType))
                    count += 1;
                else
                    break;
            // Determine whether the property has been found.
            if(count == indexes.Length)
                // Determine whether the return type can be converted to the properties type.
                if(CanConvertFrom(returnType, match[i].PropertyType))
                    return match[i];
                else
                    continue;
        }
        return null;
    }
    // Determines whether type1 can be converted to type2. Check only for primitive types.
    private bool CanConvertFrom(Type type1, Type type2)
    {
        if(type1.IsPrimitive && type2.IsPrimitive)
        {
            TypeCode typeCode1 = Type.GetTypeCode(type1);
            TypeCode typeCode2 = Type.GetTypeCode(type2);
            // If both type1 and type2 have the same type, return true.
            if(typeCode1 == typeCode2)
                return true;
            // Possible conversions from Char follow.
            if(typeCode1 == TypeCode.Char)
                switch(typeCode2)
                {
                    case TypeCode.UInt16 : return true;
                    case TypeCode.UInt32 : return true;
                    case TypeCode.Int32  : return true;
                    case TypeCode.UInt64 : return true;
                    case TypeCode.Int64  : return true;
                    case TypeCode.Single : return true;
                    case TypeCode.Double : return true;
                    default              : return false;
                }
            // Possible conversions from Byte follow.
            if(typeCode1 == TypeCode.Byte)
                switch(typeCode2)
                {
                    case TypeCode.Char   : return true;
                    case TypeCode.UInt16 : return true;
                    case TypeCode.Int16  : return true;
                    case TypeCode.UInt32 : return true;
                    case TypeCode.Int32  : return true;
                    case TypeCode.UInt64 : return true;
                    case TypeCode.Int64  : return true;
                    case TypeCode.Single : return true;
                    case TypeCode.Double : return true;
                    default              : return false;
                }
            // Possible conversions from SByte follow.
            if(typeCode1 == TypeCode.SByte)
                switch(typeCode2)
                {
                    case TypeCode.Int16  : return true;
                    case TypeCode.Int32  : return true;
                    case TypeCode.Int64  : return true;
                    case TypeCode.Single : return true;
                    case TypeCode.Double : return true;
                    default              : return false;
                }
            // Possible conversions from UInt16 follow.
            if(typeCode1 == TypeCode.UInt16)
                switch(typeCode2)
                {
                    case TypeCode.UInt32 : return true;
                    case TypeCode.Int32  : return true;
                    case TypeCode.UInt64 : return true;
                    case TypeCode.Int64  : return true;
                    case TypeCode.Single : return true;
                    case TypeCode.Double : return true;
                    default              : return false;
                }
            // Possible conversions from Int16 follow.
            if(typeCode1 == TypeCode.Int16)
                switch(typeCode2)
                {
                    case TypeCode.Int32  : return true;
                    case TypeCode.Int64  : return true;
                    case TypeCode.Single : return true;
                    case TypeCode.Double : return true;
                    default              : return false;
                }
            // Possible conversions from UInt32 follow.
            if(typeCode1 == TypeCode.UInt32)
                switch(typeCode2)
                {
                    case TypeCode.UInt64 : return true;
                    case TypeCode.Int64  : return true;
                    case TypeCode.Single : return true;
                    case TypeCode.Double : return true;
                    default              : return false;
                }
            // Possible conversions from Int32 follow.
            if(typeCode1 == TypeCode.Int32)
                switch(typeCode2)
                {
                    case TypeCode.Int64  : return true;
                    case TypeCode.Single : return true;
                    case TypeCode.Double : return true;
                    default              : return false;
                }
            // Possible conversions from UInt64 follow.
            if(typeCode1 == TypeCode.UInt64)
                switch(typeCode2)
                {
                    case TypeCode.Single : return true;
                    case TypeCode.Double : return true;
                    default              : return false;
                }
            // Possible conversions from Int64 follow.
            if(typeCode1 == TypeCode.Int64)
                switch(typeCode2)
                {
                    case TypeCode.Single : return true;
                    case TypeCode.Double : return true;
                    default              : return false;
                }
            // Possible conversions from Single follow.
            if(typeCode1 == TypeCode.Single)
                switch(typeCode2)
                {
                    case TypeCode.Double : return true;
                    default              : return false;
                }
        }
        return false;
    }
}
public class MyClass1
{
    public short myFieldB;
    public int myFieldA;
    public void MyMethod(long i, char k)
    {
        Console.WriteLine("\nThis is MyMethod(long i, char k)");
    }
    public void MyMethod(long i, long j)
    {
        Console.WriteLine("\nThis is MyMethod(long i, long j)");
    }
}
public class Binder_Example
{
    public static void Main()
    {
        // Get the type of MyClass1.
        Type myType = typeof(MyClass1);
        // Get the instance of MyClass1.
        MyClass1 myInstance = new MyClass1();
        Console.WriteLine("\nDisplaying the results of using the MyBinder binder.\n");
        // Get the method information for MyMethod.
        MethodInfo myMethod = myType.GetMethod("MyMethod", BindingFlags.Public | BindingFlags.Instance,
            new MyBinder(), new Type[] {typeof(short), typeof(short)}, null);
        Console.WriteLine(myMethod);
        // Invoke MyMethod.
        myMethod.Invoke(myInstance, BindingFlags.InvokeMethod, new MyBinder(), new Object[] {(int)32, (int)32}, CultureInfo.CurrentCulture);
    }
}

Комментарии

Этот метод динамически вызывает метод, отраженный этим экземпляром в obj, и передает указанные параметры. Если метод является статическим obj , параметр игнорируется. Для нестатических методов должен быть экземпляром класса, obj который наследует или объявляет метод и должен иметь тот же тип, что и этот класс. Если метод не имеет параметров, значение должно иметь значение parametersnull. В противном случае число, тип и порядок элементов в parameters должны совпадать с числом, типом и порядком параметров метода, отражаемым этим экземпляром.

Вы не можете пропускать необязательные параметры в вызовах .Invoke Чтобы вызвать метод и опустить необязательные параметры, вызовите Type.InvokeMember вместо него .

Примечание

Если эта перегрузка метода используется для вызова конструктора экземпляра, объект, предоставленный для obj , повторно инициализируется, то есть выполняются все инициализаторы экземпляров. Возвращается значение null. Если вызывается конструктор класса, класс повторно инициализируется; то есть выполняются все инициализаторы классов. Возвращается значение null.

Для примитивных параметров, передаваемых по значению, выполняется обычное расширение (например, Int16 —> Int32). Для параметров ссылок, передаваемых по значению, допускается обычное расширение ссылок (производный класс к базовому классу и базовый класс к типу интерфейса). Однако для примитивных параметров, передаваемых по ссылке, типы должны точно соответствовать. Для параметров ссылки по ссылке по-прежнему применяется обычное расширение.

Например, если метод, отражаемый этим экземпляром, объявлен как public boolean Compare(String a, String b), то parameters должен быть массивом с длиной Objects 2, например parameters[0] = new Object("SomeString1") and parameters[1] = new Object("SomeString2").

Если параметр текущего метода является типом значения, а соответствующий аргумент в parametersnull, среда выполнения передает инициализированный с нуля экземпляр типа значения.

Отражение использует динамический поиск методов при вызове виртуальных методов. Например, предположим, что класс B наследует от класса A и реализует виртуальный метод С именем M. Теперь предположим, что у вас есть MethodInfo объект , представляющий M в классе A. Если метод используется Invoke для вызова M для объекта типа B, отражение будет использовать реализацию, заданную классом B. Даже если объект типа B приведен к A, используется реализация класса B (см. пример кода ниже).

С другой стороны, если метод не является виртуальным, отражение будет использовать реализацию, заданную типом, из которого MethodInfo был получен , независимо от типа объекта, переданного в качестве целевого объекта.

Ограничения доступа игнорируются для полностью доверенного кода. То есть к закрытым конструкторам, методам, полям и свойствам можно обращаться и вызывать с помощью отражения всякий раз, когда код является полностью доверенным.

Если вызываемый метод создает исключение, Exception.GetBaseException метод возвращает исходное исключение.

Примечание

Начиная с .NET Framework 2.0, этот метод можно использовать для доступа к не открытым членам, если вызывающему объекту предоставлен ReflectionPermissionReflectionPermissionFlag.RestrictedMemberAccess флаг, и если набор разрешений для не являющихся открытыми членами ограничен набором разрешений вызывающего объекта или его подмножеством. (См. раздел Вопросы безопасности для отражения.) Чтобы использовать эту функцию, приложение должно быть предназначено для .NET Framework 3.5 или более поздней версии.

См. также раздел

Применяется к

.NET 10 и другие версии
Продукт Версии
.NET Core 2.0, Core 2.1, Core 2.2, Core 3.0, Core 3.1, 5, 6, 7, 8, 9, 10
.NET Framework 1.1, 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, 2.1