C# 反射(转)

什么是反射

Reflection,中文翻译为反射。
        这是.Net中获取运行时类型信息的方式,.Net的应用程序由几个部分:‘程序集(Assembly)’、‘模块(Module)’、‘类型 (class)’组成,而反射提供一种编程的方式,让程序员可以在程序运行期获得这几个组成部分的相关信息,例如:

Assembly类可以获得正在运行的装配件信息,也可以动态的加载装配件,以及在装配件中查找类型信息,并创建该类型的实例。
Type类可以获得对象的类型信息,此信息包含对象的所有要素:方法、构造器、属性等等,通过Type类可以得到这些要素的信息,并且调用之。
MethodInfo包含方法的信息,通过这个类可以得到方法的名称、参数、返回值等,并且可以调用之。
诸如此类,还有FieldInfo、EventInfo等等,这些类都包含在System.Reflection命名空间下。

2、命名空间与装配件的关系
        很多人对这个概念可能还是很不清晰,对于合格的.Net程序员,有必要对这点进行澄清。
        命名空间类似与Java的包,但又不完全等同,因为Java的包必须按照目录结构来放置,命名空间则不需要。

装配件是.Net应用程序执行的最小单位,编译出来的.dll、.exe都是装配件。

装配件和命名空间的关系不是一一对应,也不互相包含,一个装配件里面可以有多个命名空间,一个命名空间也可以在多个装配件中存在,这样说可能有点模糊,举个例子:
首先我们建立一个类库,将它生成为HelloWorld.dll,

using System;

 namespace Webtest
 ...{

    public interface interface1
     ...{
          int add();
     
     }
     public class ReflectTest:interface1
     ...{
         
         public String Write;
         private String Writec;

         public String Writea
         ...{
             get
             ...{
                 return Write;
             }
             set
             ...{
                 Write = value;
             }
         
         }

         private String Writeb
         ...{
             get
             ...{
                 return Writec;
             }
             set
             ...{
                 Writec = value;
             }

         }

          public ReflectTest()
          ...{
              this.Write = "Write";
              this.Writec = "Writec";
          }

         public ReflectTest(string str1,string str2)
         ...{
             this.Write = str1;
             this.Writec = str2;
         }

         public string WriteString(string s,int b)
         ...{
             return "欢迎您," + s + "---" + b; ;
         }

          public static string WriteName(string s)
          ...{
             return "欢迎您光临," + s;
          }

         public string WriteNoPara()
         ...{
            return "您使用的是无参数方法";
         }

         private string WritePrivate()
         ...{
             return "私有类型的方法";
         }


         public int add()
         ...{
             return 5;
         }
     }
}

然后,建立再建立一个项目引入该HelloWorld.dll,

using System;

using System.Threading;
using System.Reflection;


class Test
...{
    delegate string TestDelegate(string value,int value1);

   static void Main()
    ...{
        //Assembly t = Assembly.LoadFrom("HelloWorld.dll"); 与下面相同的效果
        Assembly t = Assembly.Load("HelloWorld");

//**********************************************************************     
       foreach (Type aaa in t.GetTypes())
       ...{
            //Console.Write(aaa.Name);   //显示该dll下所有的类
        }

//**********************************************************************
        Module[] modules = t.GetModules();

        foreach (Module module in modules)
        ...{
            //Console.WriteLine("module name:" + module.Name);//显示模块的名字本例为"HelloWorld.dll"
        }

//**********************************************************************
        Type a = typeof(Webtest.ReflectTest);//得到具体的类的类型,和下面一个效果
        //Type a = t.GetType("Webtest.ReflectTest");//
        //Console.Write(a.Name);

//**********************************************************************
        string[] bb =...{ "aaaa", "bbbbb" };
        object obj = Activator.CreateInstance(a,bb); //创建该类的实例,后面的bb为有参构造函数的参数
        //object obj = t.CreateInstance("Webtest.ReflectTest");//与上面方法相同

//**********************************************************************        
        MethodInfo[] miArr = a.GetMethods();
        foreach (MethodInfo mi0 in miArr)
       ...{
            //Console.Write(mi0.Name);  //显示所有的共有方法
       }

//**********************************************************************
        MethodInfo mi = a.GetMethod("WriteString");//显示具体的方法
        object[] aa=...{"使用的是带有参数的非静态方法",2};
        string s = (string)mi.Invoke(obj,aa); //带参数方法的调用

        MethodInfo mi1 = a.GetMethod("WriteName");
        String[] aa1 =...{"使用的是静态方法"};
        string s1 = (string)mi1.Invoke(null, aa1); //静态方法的调用

        MethodInfo mi2 = a.GetMethod("WriteNoPara");
        string s2 = (string)mi2.Invoke(obj, null); //不带参数的方法调用

        MethodInfo mi3 = a.GetMethod("WritePrivate",BindingFlags.Instance | BindingFlags.NonPublic);
        string s3 = (string)mi3.Invoke(obj, null); //私有类型方法调用

        //Console.Write(s3);

//**********************************************************************
        PropertyInfo[] piArr = a.GetProperties(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);
        foreach (PropertyInfo pi in piArr)
        ...{
         //Console.Write(pi.Name);  //显示所有的属性
        }

//**********************************************************************
        PropertyInfo pi1=a.GetProperty("Writea");
        //pi1.SetValue(obj, "Writea", null);
        //Console.Write(pi1.GetValue(obj,null));

        PropertyInfo pi2 = a.GetProperty("Writeb", BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public);
        pi2.SetValue(obj, "Writeb", null);
        //Console.Write(pi2.GetValue(obj, null));

        FieldInfo fi1 = a.GetField("Write");
        //Console.Write(fi1.GetValue(obj));

//**********************************************************************
        ConstructorInfo[] ci1 = a.GetConstructors();
        foreach (ConstructorInfo ci in ci1)
        ...{
            //Console.Write(ci.ToString()); //获得构造函数的形式
        }

        ConstructorInfo asCI = a.GetConstructor(new Type[] ...{ typeof(string), typeof(string) });
        //Console.Write(asCI.ToString());

//**********************************************************************
        Webtest.interface1 obj1 = (Webtest.interface1)t.CreateInstance("Webtest.ReflectTest");
        Webtest.ReflectTest obj2 = (Webtest.ReflectTest)t.CreateInstance("Webtest.ReflectTest");
        //Console.Write(obj1.add());典型的工厂模式

//**********************************************************************
        foreach (Type tt in t.GetTypes())
        ...{
            if (tt.GetInterface("interface1")!=null)
            ...{
                Webtest.interface1 obj3 = (Webtest.interface1)Activator.CreateInstance(a);
                //Console.Write(obj3.add());
            }
        }

//**********************************************************************
        TestDelegate method = (TestDelegate)Delegate.CreateDelegate(typeof(TestDelegate), obj, "WriteString");
       //动态创建委托的简单例子
        //Console.Write(method("str1", 2));

//**********************************************************************
        ConstructorInfo asCI1 = a.GetConstructor(new Type[0]);
        Webtest.ReflectTest obj5 = (Webtest.ReflectTest)asCI1.Invoke(null);
            //通过无参构造函数实例化的方法
        //Console.Write(obj5.Writea);

        ConstructorInfo asCI2 = a.GetConstructor(new Type[] ...{ typeof(string), typeof(string) });
          //通过有参构造函数实例化的方法
        Webtest.ReflectTest obj6 = (Webtest.ReflectTest)asCI2.Invoke(bb);
        Console.Write(obj6.Writea);
//**********************************************************************

        Console.Read();
    }   
}

时间: 2024-12-23 21:28:25

C# 反射(转)的相关文章

Android小例子:使用反射机制来读取图片制作一个图片浏览器

效果图: 工程文件夹: 该例子可供于新手参考练习,如果有哪里不对的地方,望指正>-< <黑幕下的人> java代码(MainActivity.java): package com.example.imageswitchtest; import java.lang.reflect.Field; import android.app.Activity; import android.os.Bundle; import android.util.Log; import android.v

.Net 反射

反射是.NET中的重要机制,通过反射,可以在运行时获得程序或程序集中每一个类型(包括类.结构.委托.接口和枚举等)的成员和成员的信息.有了反射,即可对每一个类型了如指掌.另外我还可以直接创建对象,即使这个对象的类型在编译时还不知道.     反射的用途:    (1)使用Assembly定义和加载程序集,加载在程序集清单中列出模块,以及从此程序集中查找类型并创建该类型的实例.     (2)使用Module了解包含模块的程序集以及模块中的类等,还可以获取在模块上定义的所有全局方法或其他特定的非全

C#图解教程 第二十四章 反射和特性

反射和特性元数据和反射Type 类获取Type对象什么是特性应用特性预定义的保留的特性Obsolete(废弃)特性Conditional特性调用者信息特性DebuggerStepThrough 特性其他预定义特性有关应用特性的更多内容多个特性其他类型的目标全局特性自定义特性声明自定义特性使用特性的构造函数指定构造函数使用构造函数构造函数中的位置参数和命名参数限制特性的使用自定义特性的最佳实践访问特性使用IsDefined方法使用GetCustomAttributes方法 Note 类的元数据包含

Java反射

1. 介绍 反射是一种能够在程序运行时动态访问.修改某个类中任意属性和方法的机制. 具体:对于任意一个类,都能够知道这个类的所有属性和方法对于任意一个对象,都能够调用它的任意一个方法和属性 在运行时,当加载完类之后,JVM在堆内存中会自动产生一个Class类型的对象,这个对象包含了完整的类的结构信息 这个Class对象就像一面镜子,透过这个镜子看到类的结构 那么,如何得到这个Class对象呢?以下可否 Class c = new Class(); 答案是不行的,因为Class的构造函数定义为私有

深入理解Java:类加载机制及反射

一.Java类加载机制 1.概述 Class文件由类装载器装载后,在JVM中将形成一份描述Class结构的元信息对象,通过该元信息对象可以获知Class的结构信息:如构造函数,属性和方法等,Java允许用户借由这个Class相关的元信息对象间接调用Class对象的功能. 虚拟机把描述类的数据从class文件加载到内存,并对数据进行校验,转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制. 2.工作机制 类装载器就是寻找类的字节码文件,并构造出类在JVM内部表示

通过反射了解集合泛型的本质

通过反射了解集合泛型的本质 import java.lang.reflect.Method; import java.util.ArrayList; /** * 通过反射了解集合泛型的本质 * @author shm * */ public class MethodDemo02 { public static void main(String[] args) { ArrayList list = new ArrayList(); list.add("hello"); list.add(

Java 反射详解

反射反射,程序员的快乐,今天你快乐了吗?如果你不快乐,没关系,接下来让你快乐起来! 一.什么是反射? 通过百度百科我们可以知道,Java反射就是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意方法和属性:并且能改变它的属性.而这也是Java被视为动态(或准动态,为啥要说是准动态,因为一般而言的动态语言定义是程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言.从这个观点看,Perl,Python,Ruby是动态语言,C++,Java,C

java中的反射机制和javaBean

反射 反射:就是通过一个类加载进方法区时加载到栈内存中的Class字节码文件对这个类进行解剖 通过反射可以获取到一个类的构造方法,成员方法,成员变量 反射将一个类的各个部分映射成相应的类 反射获取构造方法 Class类中方法 Constructor<?>[] getConstructors() 返回当前字节码文件对象的所有public修饰的构造方法 Constructor<T> getConstructor(Class<?>...parameterTypes)返回指定了

反射机制2,Class类的使用

class是反射源头,不光可以取得对象所在类信息,也可直接通过class类的方法进行对象的实例化操作. 使用关键字new为对象实例化.如果已经实例化好了class对象,就可以通过class类中提供的newInstance()操作 public T newInstance() throws InstantiationException, IllegalAccessException 来个例子: package 类集; class Person{ private String name ; // n

python 反射

1. 反射 实例:伪造web框架中的路由系统 利用反射导入模块 obj = __import__('commons') obj = __import__('lib.' + 'commons',fromlist = True) 导入模块不在当前目录下,需要设置fromlist参数为真, 才能拼接模块路径 利用反射操作对象成员(属性) 通过传入字符串,操作(检查/获取/删除/设置)对象成员(属性) hasattr(对象名,字符串),如果对象中存在名为字符串的属性(成员),返回True,否则返回Fal