Учебники

C # — Отражение

Объекты отражения используются для получения информации о типе во время выполнения. Классы, предоставляющие доступ к метаданным работающей программы, находятся в пространстве имен System.Reflection .

Пространство имен System.Reflection содержит классы, которые позволяют получать информацию о приложении и динамически добавлять типы, значения и объекты в приложение.

Приложения отражения

Отражение имеет следующие приложения —

  • Это позволяет просматривать информацию об атрибутах во время выполнения.

  • Это позволяет исследовать различные типы в сборке и создавать экземпляры этих типов.

  • Это позволяет позднее связывание с методами и свойствами

  • Это позволяет создавать новые типы во время выполнения, а затем выполняет некоторые задачи, используя эти типы.

Это позволяет просматривать информацию об атрибутах во время выполнения.

Это позволяет исследовать различные типы в сборке и создавать экземпляры этих типов.

Это позволяет позднее связывание с методами и свойствами

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

Просмотр метаданных

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

Объект MemberInfo класса System.Reflection должен быть инициализирован для обнаружения атрибутов, связанных с классом. Для этого вы определяете объект целевого класса, как —

System.Reflection.MemberInfo info = typeof(MyClass);

Следующая программа демонстрирует это —

using System;

[AttributeUsage(AttributeTargets.All)]
public class HelpAttribute : System.Attribute {
   public readonly string Url;
   
   public string Topic   // Topic is a named parameter {
      get {
         return topic;
      }
      set {
         topic = value;
      }
   }
   public HelpAttribute(string url)   // url is a positional parameter {
      this.Url = url;
   }
   private string topic;
}

[HelpAttribute("Information on the class MyClass")]
class MyClass {

}

namespace AttributeAppl {
   class Program {
      static void Main(string[] args) {
         System.Reflection.MemberInfo info = typeof(MyClass);
         object[] attributes = info.GetCustomAttributes(true);
         
         for (int i = 0; i < attributes.Length; i++) {
            System.Console.WriteLine(attributes[i]);
         }
         Console.ReadKey();
      }
   }
}

Когда он скомпилирован и запущен, он отображает имя пользовательских атрибутов, прикрепленных к классу MyClass

HelpAttribute

пример

В этом примере мы используем атрибут DeBugInfo, созданный в предыдущей главе, и используем отражение для чтения метаданных в классе Rectangle .

Live Demo

using System;
using System.Reflection;

namespace BugFixApplication {
   //a custom attribute BugFix to be assigned to a class and its members
   [AttributeUsage(
      AttributeTargets.Class |
      AttributeTargets.Constructor |
      AttributeTargets.Field |
      AttributeTargets.Method |
      AttributeTargets.Property,
      AllowMultiple = true)]

   public class DeBugInfo : System.Attribute {
      private int bugNo;
      private string developer;
      private string lastReview;
      public string message;
      
      public DeBugInfo(int bg, string dev, string d) {
         this.bugNo = bg;
         this.developer = dev;
         this.lastReview = d;
      }
      public int BugNo {
         get {
            return bugNo;
         }
      }
      public string Developer {
         get {
            return developer;
         }
      }
      public string LastReview {
         get {
            return lastReview;
         }
      }
      public string Message {
         get {
            return message;
         }
         set {
            message = value;
         }
      }
   }
   [DeBugInfo(45, "Zara Ali", "12/8/2012", Message = "Return type mismatch")]
   [DeBugInfo(49, "Nuha Ali", "10/10/2012", Message = "Unused variable")]
   
   class Rectangle {
      //member variables
      protected double length;
      protected double width;
      
      public Rectangle(double l, double w) {
         length = l;
         width = w;
      }
      [DeBugInfo(55, "Zara Ali", "19/10/2012", Message = "Return type mismatch")]
      public double GetArea() {
         return length * width;
      }
      [DeBugInfo(56, "Zara Ali", "19/10/2012")]
      public void Display() {
         Console.WriteLine("Length: {0}", length);
         Console.WriteLine("Width: {0}", width);
         Console.WriteLine("Area: {0}", GetArea());
      }
   }//end class Rectangle
   
   class ExecuteRectangle {
      static void Main(string[] args) {
         Rectangle r = new Rectangle(4.5, 7.5);
         r.Display();
         Type type = typeof(Rectangle);
         
         //iterating through the attribtues of the Rectangle class
         foreach (Object attributes in type.GetCustomAttributes(false)) {
            DeBugInfo dbi = (DeBugInfo)attributes;
            
            if (null != dbi) {
               Console.WriteLine("Bug no: {0}", dbi.BugNo);
               Console.WriteLine("Developer: {0}", dbi.Developer);
               Console.WriteLine("Last Reviewed: {0}", dbi.LastReview);
               Console.WriteLine("Remarks: {0}", dbi.Message);
            }
         }
         
         //iterating through the method attribtues
         foreach (MethodInfo m in type.GetMethods()) {
            
            foreach (Attribute a in m.GetCustomAttributes(true)) {
               DeBugInfo dbi = (DeBugInfo)a;
               
               if (null != dbi) {
                  Console.WriteLine("Bug no: {0}, for Method: {1}", dbi.BugNo, m.Name);
                  Console.WriteLine("Developer: {0}", dbi.Developer);
                  Console.WriteLine("Last Reviewed: {0}", dbi.LastReview);
                  Console.WriteLine("Remarks: {0}", dbi.Message);
               }
            }
         }
         Console.ReadLine();
      }
   }
}

Когда приведенный выше код компилируется и выполняется, он дает следующий результат —