在开始正题之前,先补充一点前面的内容。
    在方法中,如果要引用方法参数,前面的示例中,老周使用的是 CodeVariableReferenceExpression 类,它用于引用变量,也适用于引用方法参数。除了这个类,还可以使用 CodeArgumentReferenceExpression 类,这个类是专为方法参数引用而设计,其实用起来也和变量引用一样。请看看下面的例子。

    1. CodeMemberMethod m = new CodeMemberMethod();
    2. m.Name = "Test";
    3. // 输出参数
    4. CodeParameterDeclarationExpression p = new CodeParameterDeclarationExpression(typeof(int), "a");
    5. p.Direction = FieldDirection.Out;
    6. m.Parameters.Add(p);
    7. // 赋值语句
    8. CodeAssignStatement ass = new CodeAssignStatement();
    9. ass.Left = new CodeArgumentReferenceExpression(p.Name);
    10. ass.Right = new CodePrimitiveExpression(100000);
    11. m.Statements.Add(ass);
    12. CodeDomProvider provider = CodeDomProvider.CreateProvider("cs");
    13. CodeGeneratorOptions opt = new CodeGeneratorOptions
    14. {
    15. BracingStyle = "C"
    16. };
    17. provider.GenerateCodeFromMember(m, Console.Out, opt);

    Test 方法带有一个输出参数a,int类型,方法体中给参数a赋值。上面代码执行后,生成的代码如下图所示。
    image.png
    ============================================
    好,进入主题,今天咱们来聊一聊“索引”这玩意儿。通常,使用索引的类型有:数组、List、哈希表/字典等。使用索引的类型成员有索引器。索引器可能不多见,其实跟属性很像。
    先来看看数组的初始化方法。初始化数组是一类表达式,主要用到 CodeArrayCreateExpression 类。
    看一个简单的例子。

    1. CodeArrayCreateExpression ce = new CodeArrayCreateExpression();
    2. ce.CreateType = new CodeTypeReference(typeof(string));
    3. ce.Size = 5;
    4. CodeDomProvider p = CodeDomProvider.CreateProvider("C#");
    5. p.GenerateCodeFromExpression(ce, Console.Out, null);

    CreateType 属性用来指定数组的类型,本例中类型为string,Size属性指定数组的大小。得到的代码如下图所示。
    image.png
    由于不是所有语言都支持多维数组和嵌套数组,所以,目前来说,CodeDom并不能实例化多维数组。不过也没关系,毕竟N维数组和嵌套数组很少会用得上。
    有时候,在实例化数组对象的时候就顺便对元素进行初始化,可以将初始化元素的表达式添加到 Initializers 集合中。请看下面例子。

    1. CodeArrayCreateExpression arrce = new CodeArrayCreateExpression();
    2. arrce.CreateType = new CodeTypeReference(typeof(int));
    3. // 初始化元素
    4. arrce.Initializers.Add(new CodePrimitiveExpression(1));
    5. arrce.Initializers.Add(new CodePrimitiveExpression(3));
    6. arrce.Initializers.Add(new CodePrimitiveExpression(5));
    7. // 声明变量
    8. CodeVariableDeclarationStatement vd = new CodeVariableDeclarationStatement();
    9. vd.Name = "x";
    10. vd.Type = new CodeTypeReference(typeof(int[]));
    11. vd.InitExpression = arrce;

    最后使用 CodeVariableDeclarationStatement 产生一句完整的变量声明语句,初始化变量时就用CodeArrayCreateExpression表达式。
    生成的代码如下图所示。
    image.png
    要访问某个数组变量的指定索引处的值,可以用 CodeArrayIndexerExpression 类。

    1. CodeArrayIndexerExpression aiexp = new CodeArrayIndexerExpression();
    2. aiexp.TargetObject = new CodeVariableReferenceExpression("x");
    3. aiexp.Indices.Add(new CodePrimitiveExpression(0));

    TargetObject属性用来设置对数组变量的引用,Indices集合用来添加索引引用表达式,数组的索引都是整数,所以,应当用以下表达式:
    new CodePrimitiveExpression(<整数值>)
    上面示例生成的代码如下图所示。
    image.png
    =================================================
    下面咱们看看索引器。
    访问索引器最典型的一个应用是字典。下面例子将生成三个语句,其中,第一句是声明被初始化一个字典变量,第二句和第三句都是向字典变量添加元素。

    1. // 声明并初始化字典变量
    2. CodeVariableDeclarationStatement vd = new CodeVariableDeclarationStatement();
    3. vd.Type = new CodeTypeReference(typeof(Dictionary<string, string>));
    4. vd.Name = "dic";
    5. vd.InitExpression = new CodeObjectCreateExpression(typeof(Dictionary<string, string>));
    6. // 给字典对象添加元素
    7. // 左边:索引引用
    8. // 右边:值
    9. CodeAssignStatement ass = new CodeAssignStatement();
    10. ass.Left = new CodeIndexerExpression(new CodeVariableReferenceExpression(vd.Name), new CodePrimitiveExpression("a"));
    11. ass.Right = new CodePrimitiveExpression("cake");
    12. CodeAssignStatement ass1 = new CodeAssignStatement();
    13. ass1.Left = new CodeIndexerExpression(new CodeVariableReferenceExpression(vd.Name), new CodePrimitiveExpression("b"));
    14. ass1.Right = new CodePrimitiveExpression("bird");
    15. CodeDomProvider prd = CodeDomProvider.CreateProvider("cs");
    16. prd.GenerateCodeFromStatement(vd, Console.Out, null);
    17. prd.GenerateCodeFromStatement(ass, Console.Out, null);
    18. prd.GenerateCodeFromStatement(ass1, Console.Out, null);

    引用某个实例的索引器,应使用 CodeIndexerExpression 类。TargetObject属性用来指定要引用的对象,通常是变量引用,Indices属性是索引集合,用来指定要访问的索引。这些属性的值可以直接向CodeIndexerExpression类的构造函数传递。
    本例生成的代码如下。

    1. System.Collections.Generic.Dictionary<string, string> dic = new System.Collections.Generic.Dictionary<string, string>();
    2. dic["a"] = "cake";
    3. dic["b"] = "bird";

    如果要给自定义的类型声明索引器,要用 CodeMemberProperty 类,因为索引器与属性相似。还是用例子说话吧。

    1. CodeTypeDeclaration td = new CodeTypeDeclaration("Sample");
    2. td.Attributes = MemberAttributes.Public;
    3. // 索引器
    4. CodeMemberProperty mb = new CodeMemberProperty();
    5. mb.Type = new CodeTypeReference(typeof(string));
    6. mb.Name = "item";
    7. mb.Attributes = MemberAttributes.Public | MemberAttributes.Final;
    8. // 索引参数
    9. mb.Parameters.Add(new CodeParameterDeclarationExpression(typeof(int), "k"));
    10. td.Members.Add(mb);
    11. CodeDomProvider provider = CodeDomProvider.CreateProvider("cs");
    12. provider.GenerateCodeFromType(td, Console.Out, null);

    这里有个地方要注意,就是索引器成员的名字,为了兼容各种语言,较合适的做法是把它命名为“item”或“Item”(不分大小写),这样一来,生成C#代码时,就能够生成 this[int k] 这样的语法,只有这样的语法才能被认为是索引器。
    生成的代码如下图所示。
    image.png