Г. Шилдт - Полный справочник по C# (1160789), страница 85
Текст из файла (страница 85)
Для static-методов параметр ob должен содержать значение n u l l . Любые аргументы, которые необходимо передать вызываемому методу, указываются в массивеargs. Если метод вызывается без аргументов, параметр args должен иметь n u l l значение. При этом длина массива args должна совпадать с количеством аргументов,передаваемых методу. Следовательно, если необходимо передать два аргумента, массив args должен состоять из двух элементов, а не, например, из трех или четырех.Глава 17. Динамическая идентификация типов, отражение и атрибуты459Для вызова нужного метода достаточно вызвать метод invoke () для экземпляракласса Methodlnfo, полученного в результате вызова метода GetMethods ().
Эта процедура демонстрируется следующей программой:// Вызов методов с использованием средства отражения.using System;using System.Reflection;class MyClass {int x;int y;public MyClass(int i, int j) {x = i;У = j;}public int sum() {return x+y;}public bool isBetween(int i) {if((x < i) && (i < y)) return true;else return false;}public void set (int a, int b) {Console.Write("Внутри метода set(int, int). " ) ; .x = a;У = b;show();}// Перегруженный метод set.public void set(double a, double b) {Console.Write("Внутри метода set(double, double).
" ) ;x = (int) a;у = (int) b;show();p u b l i c v o i d show() {Console.WriteLine("Значение x:{0},значение у:{1}",x,у);class InvokeMethDemo {public static void Main() {Type t = typeof(MyClass);MyClass reflectOb = new MyClass(10, 20);int val;Console.WriteLine("Вызов методов, определенных вt.Name);Console.WriteLine() ;M e t h o d l n f o [ ] mi = t . G e t M e t h o d s ( ) ;460Часть I.
Язык С#// Вызываем каждый метод,foreach(Methodlnfo m in mi) {// Получаем параметры.Parameterlnfо[] pi = m.GetParameters();if(m.Name.CompareTo("set")==0 &&pi[0].ParameterType == typeof(int)) {object [] args = new object[2];args[0j = 9;args[l] = 18;m.Invoke(reflectOb, args);}else if(m.Name.CompareTo("set")==0 &&pi[0].ParameterType — typeof(double)) {object[] args = new object[2];args[0] = 1.12;args[l] = 23.4;m.Invoke(reflectOb, args);}else if(m.Name.CompareTo("sum")==0) {val = (int) m.Invoke(reflectOb, null);Console.WriteLine("Результат вызова метода sum равен " + val)}else if(m.Name.CompareTo("isBetween")==0) {object[] args = new object[1];args[0J = 14;if((bool) m.Invoke(reflectOb, args))Console.WriteLine("14 находится между х и у .
" ) ;}else if(m.Name.CompareTo("show")==0) {m.Invoke(reflectOb, null);Результаты выполнения этой программы таковыТ"Вызов методов, определенных в MyClassРезультат вызова метода sum равен 3014 находится между х и у.Внутри метода s e t ( i n t , i n t ) . Значение х: 9, значение у: 18Внутри метода set(double, double). Значение х: 1, значение у: 23Значение х: 1, значение у: 23Обратите внимание на то, как организуется вызов методов. Сначала получаем список методов. Затем в цикле foreach извлекаем информацию о параметрах каждогометода. После этого, используя последовательность i f /else-инструкций, вызываемкаждый метод с соответствующим количеством параметров определенного типа. Особое внимание обратите на способ вызова перегруженного метода s e t ():if(m.Name.CompareTo("set")==0 &&pi[0].ParameterType == typeof(int)) {object[] args = new object[2];args[0] = 9;args[l] = 18;m.Invoke(reflectOb, args);Глава 17. Динамическая идентификация типов, отражение и атрибуты461e l s e if(m.Name.CompareTo("set")==0 &&p i [ 0 ] .
P a r a m e t e r T y p e — typeof(double))o b j e c t [ ] a r g s = new o b j e c t [ 2 ] ;args[0] = 1.12;a r g s [ l ] = 23.4;m.Invoke(reflectOb,args);{Если имя метода совпадает со строкой s e t , проверяется тип первого параметра,чтобы определить версию метода s e t ( ) . Если окажется, что рассматривается версияs e t ( i n t , i n t ) , в массив a r g s зафужаются int-аргументы и вызывается методs e t ( ) . В противном случае для вызова метода s e t ( ) используются аргументы типаdouble.Получение конструкторов типаВ предыдущем примере продемонстрирована возможность вызова методов с использованием средстве отражения, однако такой подход не имеет преимуществ посравнению с непосредственным вызовом методов (в данном случае класса MyClass),поскольку объект типа MyClass создается явным образом.
Другими словами, прощевызывать эти методы обычным способом. Однако мощь отражения начинает проявляться в тех случаях, когда объект создается динамически во время работы программы. Для этого нужно сначала получить список конструкторов. Затем создать экземпляр типа, вызвав один из конструкторов. Этот механизм позволяет реализовать объектлюбого типа во время работы программы, не называя его в инструкции объявления.Чтобы получить конструкторы типа, вызовите метод G e t C o n s t r u c t o r s () для объекта класса Туре. Один из наиболее употребительных форматов его вызова таков:Constructorlnfо[] GetConstructors()Он возвращает массив объектов типа C o n s t r u c t o r l n f o , которые описывают этиконструкторы.Класс C o n s t r u c t o r l n f o выведен из абстрактного класса MethodBase, которыйявляется производным от класса Member Info.
Класс C o n s t r u c t o r l n f o определяеттакже собственные члены. Из них нас интересует прежде всего методGetParameters ( ) , который возвращает список параметров, связанных с конструктором. Он работает подобно методу GetParameters ( ) , определенному в описанномвыше классе Methodlnf о.Получив информацию о конструкторе, можно с его помощью создать объект, вызвав метод Invoke ( ) , определенный в классе C o n s t r u c t o r l n f o . Формат вызова метода Invoke () в этом случае таков:object Invoke(object[]args)Любые аргументы, которые необходимо передать конструктору, задаются с помощью массива args.
Если конструктор вызывается без аргументов, параметр args должен иметь null-значение. При этом длина массива args должна в точности совпадать с количеством аргументов. Метод invoke () возвращает ссылку на создаваемыйобъект.В следующей программе демонстрируется использование средства отражения длясоздания экземпляра класса MyClass:// Создание объекта с помощью средства отражения.using System;using System.Reflection;462Часть I.
Язык С#class MyClass {int x;int y;public MyClass(int i) {Console.WriteLine("Создание объекта по формату MyClass(int). " ) ;x = у = i;public MyClass(int i, int j) {Console.WriteLine("Создание объекта по формату MyClass(int, int). " ) ;x = i;у = j;show();public int sum() {return x+y;public bool isBetween(int i) {if((x < i) && (i < y)) return true;else return false;public void set(int a, int b) {Console.Write("Внутри метода set(int, int).
" ) ;x = a;У = b;show () ;// Перегруженный метод set().public void set(double a, double b) {Console.Write("Внутри метода set(double, double). ") ;x = (int) a;у = (int) b;show();public void show() {Console.WriteLine("Значение х: {0}, значение у: {1}", x, у)c l a s s InvokeConsDemo {p u b l i c s t a t i c void Main() {Type t = typeof(MyClass);int val;// Получаем информацию о конструкторах.Constructorinfо[] ci = t.GetConstructors();Console.WriteLine("Имеются следующие конструкторы: " ) ;Глава 17. Динамическая идентификация типов, отражение и атрибуты463foreach(Constructorinfо с in ci) {// Отображаем тип возвращаемого значения и имя.Console.Write("" + t.Name + " ( " ) ;// Отображаем параметры.Parameterlnfо[] pi = с.GetParameters();for(int i=0; i < pi.Length;Console.Write(pi[i].ParameterType.Name +" " + pi[i].Name);if(i+l < pi.Length) Console.Write(", " ) ;Console.WriteLine(")");}Console.WriteLine();// Находим подходящий конструктор,int x;for(x=0; x < ci.Length; x++) {Parameterlnfо[] pi = ci[x].GetParameters();if(pi.Length == 2) break;if(x == ci.Length) {Console.WriteLine("Подходящий конструктор не найден.");return;}elseConsole.WriteLine("Найден конструктор с двумя параметрами.\п");// Создаем объект.object[] consargs = new object[2];consargs[0] = 10;consargs [1] = 20;object reflectOb = ci[x].Invoke(consargs);Console.WriteLine("ХпВызов методов для объекта reflectOb.");Console.WriteLine();Methodlnfo[] mi = t.GetMethods();// Вызываем каждый метод.foreach(Methodinfо m in mi) {// Получаем параметры.Parameterlnfо[] pi = m.GetParameters();if(m.Name.CompareTo("set")==0 &&pi[0].ParameterType == typeof(int)) {// Это метод set(int, int).object[] args = new object[2];args[0] = 9;args [1] = 18;m.Invoke(reflectOb, args);464Часть I.
Язык С#else if (m.Name.CompareTo ("set")—0 &&pi[0].ParameterType == typeof(double)) {// Это метод set(double, double).object[] args = new object[2];args[0] = 1.12;args[l] = 23.4;m.Invoke(reflectOb, args);}else if(m.Name.CompareTo("sum")==0) {val = (int) m.Invoke(reflectOb, null);Console.WriteLine("Результат выполнения метода sum() равен " + val);}else if(m.Name.CompareTo("isBetween")==0) {object[] args = new object[1];args[0] = 14;if((bool) m.Invoke(reflectOb, args))Console.WriteLine("14 находится между х и у .
" ) ;}else if(m.Name.CompareTo("show")==0) {m.Invoke(reflectOb, null);Результаты выполнения этой программы таковы:Имеются следующие конструкторы:MyClass(Int32 i)MyClass(Int32 i , Int32 j)Найден конструктор с двумя параметрами.Создание объекта по формату MyClass(int,Значение х: 10, значение у: 20Вызов методов для объектаint).reflectOb.Результат выполнения метода sum() равен 3014 находится между х и у.Внутри метода s e t ( i n t , i n t ) . Значение х: 9, значение у: 18Внутри метода set(double, double). Значение х: 1, значение у: 23Значение х: 1, значение у: 23Теперь разберемся, как используется средство отражения для создания объектакласса MyClass.
Сначала с помощью следующей инструкции получаем список открытых конструкторов:I Constructorlnfо[] ci = t.GetConstructors();Затем в целях иллюстрации отображаем конструкторы, определенные в этом классе. После этого с помощью следующего кода просматриваем полученный список, чтобы найти конструктор, который принимает два аргумента:for(x=0; х < ci.Length; x++) {Parameterlnfo[] pi = ci[x].GetParameters();if(pi.Length == 2) break;|1Если нужный конструктор обнаружен (как в данном случае), создаем объект, выполнив такую последовательность инструкций:Глава 17. Динамическая идентификация типов, отражение и атрибуты465// Создаем объект.object[] consargs = new o b j e c t [ 2 ] ;consargs[0] = 10;consargs[1] = 20;object reflectOb = ci[x].Invoke(consargs);После обращения к методу Invoke () объект r e f l e c t O b будет ссылаться на объекткласса MyClass.В этом примере в целях упрощения предполагалось, что конструктор, которыйпринимает два int-аргумента, — единственный среди всех конструкторов, определенных в классе MyClass.