时间:2021-05-19
但是,如果对象是在运行时动态创建的,反射的功能就显示出来了。在这种情况下,需要首先获取一个构造函数列表,然后再调用列表中的某个构造函数,创建一个该类型的实例。通过这种机制,可以在运行时实例化任意类型的对象而不必在声明中指定。
为了获得某个类型的构造函数,需要调用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类型。但在实际的应用程序中,必须检验每一个参数的类型。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
前文回顾本文档环境基于Vscode+GCC+CodeRunner关于C++的环境搭建请参考下面链接:由于本人具有C#开发经验,部分相同的知识就不再赘述了。只列一
互联网时代所有的企业都需要网站,不过我们是将网站外部给第三方的建站公司还是自己建立,关于网站前的基础知识自己一定要了解清楚,了解清楚网站基础知识,才能制作一个更
今天再学习一些C#的基础知识,如对IntArray进行排序:你可以在控制台应用程序中,创建一个类别,它属性和2个构造函数:SourceCodeclassAf{p
本文实例讲述了C#根据反射和特性实现ORM映射的方法。分享给大家供大家参考。具体如下:(一)关于反射什么是反射?反射就是在运行时,动态获取对象信息的方法。比如:
cisa对基础知识的要求:基础知识其实要求很多,如果说一个最小范围的话,就是cisa考试的6个章节,分别是:chapter1_信息系统审计流程(占分10%),c