C#基础知识回顾-- 反射(3)

 

获取Type对象的构造函数:

  前一篇因为篇幅问题因为篇幅太短被移除首页,反射这一块还有一篇“怎样在程序集中使用反射”,

其他没有什么可以写的了,前两篇主要是铺垫,在实际应用中,主要是最后两篇。其实写完反射,

我也考虑重新把委托重新写下,现在想想确实太过单一和简单了。我会把实际开发中委托的例子多写几个

, 以便以理解。如果这几篇C#的基础对您有所帮助的话,请点击推荐。谢谢您的支持……

  好了,不废话了,在前面例子中,由于MyClass类型的对象是显示创建的,因此使

用反射技术来调用MyClass上的方法没有任何优势--以普通的方式调用对象上的方法会简单的多。但是,

如果对象是在运行时动态创建的,反射的功能就显示出来了。在这种情况下,需要首先获取一个构造函

数列表,然后再调用列表中的某个构造函数,创建一个该类型的实例。通过这种机制,可以在运行时实例

化任意类型的对象而不必在声明中指定。

  为了获得某个类型的构造函数,需要调用Type对象上的GetConstructors()。常用形式为:

 

  ConstructorInfo[] GetConstructors()

  该方法返回一个描述构造函数的ConstructorInfo对象数组。ConstructorInfo中常用的

是GetParamters()方法,该方法返回给定构造函数的参数列表。

  一旦找到了合适的构造函数,就调用ConstructorInfo定义的Invoke()方法来创建对象:

  object Invoke(object[] args)

 

  需要传递给此方法的所有参数都在args中指定。如果不需要参数,args必须为null。另外,

 

args必须包含与参数个数相同的元素,并且实参的类型必须与形参的类型兼容。Invoke()方法返回

的是指向新构造对象的引用。

例子:

测试对象类

class MyClass
{
    int x;
    int y;

    public MyClass(int i)
    {
        Console.WriteLine("一个参数的构造函数:");
        x = y = i;
    }
    public MyClass(int i, int j)
    {
        Console.WriteLine("两个参数构造函数:");
        x = i;
        y = 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 a, int b)");
        x = a;
        y = b;

        Show();
    }

    public void Set(double a, double b)
    {
        Console.Write("函数:Set(double a, double b)");

        x = (int)a;
        y = (int)b;

        Show();
    }

    public void Show()
    {
        Console.WriteLine("x:{0},y:{1}", x, y);
    }
}

使用反射:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Reflection;
namespace Reflection
{
    class Program
    {
        static void Main(string[] args)
        {
            InvokeConsDemo();
            Console.ReadKey();
        }

        static void InvokeConsDemo()
        {

            Type t = typeof(MyClass);

            int val;

            ConstructorInfo[] ci = t.GetConstructors();

            Console.WriteLine("类构造函数如下:");

            foreach (ConstructorInfo c in ci)
            {
                Console.Write("" + t.Name + "(");

                ParameterInfo[] pi = c.GetParameters();

                for (int i = 0; i < pi.Length; i++)
                {
                    Console.Write(pi[i].ParameterType.Name + " " + pi[i].Name);

                    if (i + 1 < pi.Length) Console.Write(", ");
                }
                Console.WriteLine(") ");

            }
            Console.WriteLine();

            int x;

            for (x = 0; x < ci.Length; x++)
            {
                ParameterInfo[] pi = ci[x].GetParameters();

                if (pi.Length == 2) break;
            }

            if (x == ci.Length)
            {
                Console.WriteLine("没有找到两个参数的构造函数"); return;
            }
            else
            {
                object[] consargs = new object[2];

                consargs[0] = 10;
                consargs[1] = 20;

                object reflectOb = ci[x].Invoke(consargs);

                Console.WriteLine("用reflectOb调用方法");

                Console.WriteLine();

                MethodInfo[] mi = t.GetMethods();

                foreach (MethodInfo m in mi)
                {
                    ParameterInfo[] pi = m.GetParameters();

                    if (m.Name.CompareTo("Set") == 0 && pi[0].ParameterType == typeof(int))
                    {
                        object[] args = new object[2];

                        args[0] = 12;
                        args[1] = 7;

                        m.Invoke(reflectOb, args);
                    }
                    else if (m.Name.CompareTo("Set") == 0 && pi[0].ParameterType == typeof(double))
                    {
                        object[] args = new object[2];

                        args[0] = 1.25;
                        args[1] = 7.5;

                        m.Invoke(reflectOb, args);
                    }
                    else if (m.Name.CompareTo("Sum") == 0)
                    {
                        val = (int)m.Invoke(reflectOb, null);

                        Console.WriteLine("Sum is {0}",val);
                    }
                    else if (m.Name.CompareTo("IsBetween") == 0)
                    {
                        object[] args = new object[1];

                        args[0] = 13;

                        if ((bool)m.Invoke(reflectOb, args))
                        {
                            Console.WriteLine("13 is between x and y");
                        }

                    }
                    else if (m.Name.CompareTo("Show") == 0)
                    {
                        m.Invoke(reflectOb, null);
                    }
                }

            }
        }
    }
}

运行结果为:

 

  本例中,找到了一个两个参数的构造函数,那么使用下面的语句实例化了一个该类型的对象:

 

object reflectOb=ci[x].Invoke(consargs);

 

调用Invoke()方法后,reflectOb将引用一个MyClass类型的对象。此后,程序将执行

 

reflectOb上的方法。

 

注意:本例为了简单起见,假设了一个使用两个参数的构造函数,并且两个参数都为int类型

 

。但在实际的应用程序中,必须检验每一个参数的类型。

 

时间: 2024-09-20 07:45:00

C#基础知识回顾-- 反射(3)的相关文章

关于C#基础知识回顾--反射

其实说白了,反射就是能知道我们未知类型的类型信息这么一个东西.没什么神秘可讲!反射的核心是System.Type.System.Type包含了很多属性和方法,使用这些属性和方法可以在运行时得到类型信息   反射(reflection)是一种允许用户获得类型信息的C#特性.术语"反射"源自于它的工作方式: Type对象映射它所代表的底层对象.对Type对象进行查询可以获得(反射)与类型相关的信息.反射是一种 功能强大的机制,它允许学习和使用只在运行时才能知道的类型功能. 这些是官方定义,

C#基础知识回顾-- 反射(1)

反射(reflection)是一种允许用户获得类型信息的C#特性.术语"反射"源自于它的工作方式: Type对象映射它所代表的底层对象.对Type对象进行查询可以获得(反射)与类型相关的信息.反射是一种 功能强大的机制,它允许学习和使用只在运行时才能知道的类型功能. 这些是官方定义,其实说白了,反射就是能知道我们未知类型的类型信息这么一个东西.没什么神秘可讲! 反射的核心是System.Type.System.Type包含了很多属性和方法,使用这些属性和方法可以在运行时得到类型信息.

Java基础知识回顾--反射机制

Reflection ClassLoader的类加载机制 并非一次性加载. 需要的时候加载(运行期间动态加载). java-verbose:class可以观察类的具体加载过程. static语句块在加载后执行一次. dynamic语句块每次new新的对象都会执行,等同于构造方法中语句,用得也比较少. 首先bootstrap class loader把其他的Class loader给load进来,然后不同的class loader去load不同的class. public class TestCl

《非常网管:网络管理从入门到精通(修订版)》——第1章 网络基础知识回顾1.1 计算机网络基础

第1章 网络基础知识回顾 古语云:"练武不练功,到老一场空",学习网络的基础理论就像练功一样重要.本章主要介绍网络的基础.网络的体系结构.ISO/OSI(International Standard Organization/Open System Interconnection,国际标准化组织提出的开放系统互联)参考模型.TCP/IP(Transmission Control Protocol/Internet Protocol,传输控制协议/网际协议),其间穿插大量的实验和技巧,有

《CCNP SWITCH 300-115学习指南》——第1章 基础知识回顾

第1章 基础知识回顾CCNP SWITCH 300-115学习指南在正式进入CCNP SWITCH这门针对园区网交换技术的课程之前,我们首先快速地回顾一遍CCNA中的相关知识点并简要地介绍其中部分技术,以便于本书内容的理解.由于这里提到的所有技术都是独立存在的,如生成树或虚拟LAN(VLAN),因此本章将这些基础知识汇总到一起进行复习,并且在后续章节中将不再重复类似的基础讲解. 如果读者十分了解交换术语,并对交换技术有着基本的认识,建议跳过此章,直接从第2章开始阅读. 本章涵盖如下CCNA基础交

【JavaScript】javaScript基础知识回顾

我们首先对JavaScript的基础进行回顾. 1.浏览器的对象树 总浏大纲,还是要先看浏览器的对象树: 如1.1浏览器的对象树.png 2.window对象常用方法 alert('信息'):      消息框 prompt('提示信息',默认值): 标准输入框 confirm( ):      确认框 open( ):      打开一个新窗口 close( ):     关闭窗口 3.Form表单对象 访问表单的方式:     *  document.forms[n]     *  docu

C#基础知识回顾-- 属性与字段

今天在公交车上,突然想属性和字段到底有什么区别?很多字段属性都存在 get{}和set{} 和普通的变量没什么区别(可读可写) 我就感觉属性就是给字段一个多的选择方式,有的字段是不允许更改的.. 刚写程序时总觉得一个字段就能满足要求,何必要使用定义起来麻烦的属性 要get.set的去定义,代码量还大,真不知道他们的区别和优点例: public class myclass { public static string A private sting _B=""; public strin

C#基础知识回顾--C#遍历enum类型、获取enum项个数

C#遍历enum类型 对于enum类型: 使用foreach遍历enum类型的元素并填充combox        foreach ( HatchStyle hs1 in Enum.GetValues(typeof(HatchStyle))) { comboBox1.Items.Add(hs1.ToString()); }  获取enum项个数(使用反射) private static int NumberOfEnumValues() { return typeof(TStageEnum).Ge

Java基础知识回顾,看看你们忘了多少

1.Java中成员变量又分为实例成员变量 和类成员变量. 2.写类的目的是为了描述一类事物共有的属性 和功能.3.成员变量和局部变量的类型可以是Java中的任何一种数据类型 .4.用修饰符 static 说明的成员变量是类变量.5.变量的名字与成员变量的名字相同,则成员变量被隐藏 ,该成员变量在这个方法内暂时失效.6.用修饰符static说明的成员变量是类变量 .7.如果局部变量的名字与成员变量的名字相同,成员变量 将被隐藏.8.方法定义包括两部分:方法声明 和方法体.9.在Java中,当一个方