在开始正题之前,先补充一点前面的内容。
在方法中,如果要引用方法参数,前面的示例中,老周使用的是 CodeVariableReferenceExpression 类,它用于引用变量,也适用于引用方法参数。除了这个类,还可以使用 CodeArgumentReferenceExpression 类,这个类是专为方法参数引用而设计,其实用起来也和变量引用一样。请看看下面的例子。
CodeMemberMethod m = new CodeMemberMethod();m.Name = "Test";// 输出参数CodeParameterDeclarationExpression p = new CodeParameterDeclarationExpression(typeof(int), "a");p.Direction = FieldDirection.Out;m.Parameters.Add(p);// 赋值语句CodeAssignStatement ass = new CodeAssignStatement();ass.Left = new CodeArgumentReferenceExpression(p.Name);ass.Right = new CodePrimitiveExpression(100000);m.Statements.Add(ass);CodeDomProvider provider = CodeDomProvider.CreateProvider("cs");CodeGeneratorOptions opt = new CodeGeneratorOptions{BracingStyle = "C"};provider.GenerateCodeFromMember(m, Console.Out, opt);
Test 方法带有一个输出参数a,int类型,方法体中给参数a赋值。上面代码执行后,生成的代码如下图所示。
============================================
好,进入主题,今天咱们来聊一聊“索引”这玩意儿。通常,使用索引的类型有:数组、List、哈希表/字典等。使用索引的类型成员有索引器。索引器可能不多见,其实跟属性很像。
先来看看数组的初始化方法。初始化数组是一类表达式,主要用到 CodeArrayCreateExpression 类。
看一个简单的例子。
CodeArrayCreateExpression ce = new CodeArrayCreateExpression();ce.CreateType = new CodeTypeReference(typeof(string));ce.Size = 5;CodeDomProvider p = CodeDomProvider.CreateProvider("C#");p.GenerateCodeFromExpression(ce, Console.Out, null);
CreateType 属性用来指定数组的类型,本例中类型为string,Size属性指定数组的大小。得到的代码如下图所示。
由于不是所有语言都支持多维数组和嵌套数组,所以,目前来说,CodeDom并不能实例化多维数组。不过也没关系,毕竟N维数组和嵌套数组很少会用得上。
有时候,在实例化数组对象的时候就顺便对元素进行初始化,可以将初始化元素的表达式添加到 Initializers 集合中。请看下面例子。
CodeArrayCreateExpression arrce = new CodeArrayCreateExpression();arrce.CreateType = new CodeTypeReference(typeof(int));// 初始化元素arrce.Initializers.Add(new CodePrimitiveExpression(1));arrce.Initializers.Add(new CodePrimitiveExpression(3));arrce.Initializers.Add(new CodePrimitiveExpression(5));// 声明变量CodeVariableDeclarationStatement vd = new CodeVariableDeclarationStatement();vd.Name = "x";vd.Type = new CodeTypeReference(typeof(int[]));vd.InitExpression = arrce;
最后使用 CodeVariableDeclarationStatement 产生一句完整的变量声明语句,初始化变量时就用CodeArrayCreateExpression表达式。
生成的代码如下图所示。
要访问某个数组变量的指定索引处的值,可以用 CodeArrayIndexerExpression 类。
CodeArrayIndexerExpression aiexp = new CodeArrayIndexerExpression();aiexp.TargetObject = new CodeVariableReferenceExpression("x");aiexp.Indices.Add(new CodePrimitiveExpression(0));
TargetObject属性用来设置对数组变量的引用,Indices集合用来添加索引引用表达式,数组的索引都是整数,所以,应当用以下表达式:new CodePrimitiveExpression(<整数值>)
上面示例生成的代码如下图所示。
=================================================
下面咱们看看索引器。
访问索引器最典型的一个应用是字典。下面例子将生成三个语句,其中,第一句是声明被初始化一个字典变量,第二句和第三句都是向字典变量添加元素。
// 声明并初始化字典变量CodeVariableDeclarationStatement vd = new CodeVariableDeclarationStatement();vd.Type = new CodeTypeReference(typeof(Dictionary<string, string>));vd.Name = "dic";vd.InitExpression = new CodeObjectCreateExpression(typeof(Dictionary<string, string>));// 给字典对象添加元素// 左边:索引引用// 右边:值CodeAssignStatement ass = new CodeAssignStatement();ass.Left = new CodeIndexerExpression(new CodeVariableReferenceExpression(vd.Name), new CodePrimitiveExpression("a"));ass.Right = new CodePrimitiveExpression("cake");CodeAssignStatement ass1 = new CodeAssignStatement();ass1.Left = new CodeIndexerExpression(new CodeVariableReferenceExpression(vd.Name), new CodePrimitiveExpression("b"));ass1.Right = new CodePrimitiveExpression("bird");CodeDomProvider prd = CodeDomProvider.CreateProvider("cs");prd.GenerateCodeFromStatement(vd, Console.Out, null);prd.GenerateCodeFromStatement(ass, Console.Out, null);prd.GenerateCodeFromStatement(ass1, Console.Out, null);
引用某个实例的索引器,应使用 CodeIndexerExpression 类。TargetObject属性用来指定要引用的对象,通常是变量引用,Indices属性是索引集合,用来指定要访问的索引。这些属性的值可以直接向CodeIndexerExpression类的构造函数传递。
本例生成的代码如下。
System.Collections.Generic.Dictionary<string, string> dic = new System.Collections.Generic.Dictionary<string, string>();dic["a"] = "cake";dic["b"] = "bird";
如果要给自定义的类型声明索引器,要用 CodeMemberProperty 类,因为索引器与属性相似。还是用例子说话吧。
CodeTypeDeclaration td = new CodeTypeDeclaration("Sample");td.Attributes = MemberAttributes.Public;// 索引器CodeMemberProperty mb = new CodeMemberProperty();mb.Type = new CodeTypeReference(typeof(string));mb.Name = "item";mb.Attributes = MemberAttributes.Public | MemberAttributes.Final;// 索引参数mb.Parameters.Add(new CodeParameterDeclarationExpression(typeof(int), "k"));td.Members.Add(mb);CodeDomProvider provider = CodeDomProvider.CreateProvider("cs");provider.GenerateCodeFromType(td, Console.Out, null);
这里有个地方要注意,就是索引器成员的名字,为了兼容各种语言,较合适的做法是把它命名为“item”或“Item”(不分大小写),这样一来,生成C#代码时,就能够生成 this[int k] 这样的语法,只有这样的语法才能被认为是索引器。
生成的代码如下图所示。
