天天看点

CodeDom使用心得

CodeDom是.net提供源代码生成器,他可以通过CodeDomProvider来生成源代码,因为CodeDomProvider是CSharpCodeProvider、VBCodeProvider的抽象基类,表示CodeDomProvider是跨语言的源代码生成器,只需要提供相应的CodeProvider就可以生成相应的语言的源代码。

CodeDom在System命名空间下,CSharpCodeProvider在Microsoft.CSharp命名空间下。VBCodeProvider在Microsoft.VisualBasic命名空间下。

CodeDom里的类及相关的用法

1.CodeCompileUnit类

这个类是用来为CodeDom程序容器提供容器,可以通过他的Namespaces集合的Add方法加入一个命名空间

如:

CodeDom使用心得

CodeCompileUnit unit  =   new  CodeCompileUnit();

CodeDom使用心得

CodeNamespace theNamespace  =   new  CodeNamespace( " MyNamespace " );

CodeDom使用心得

unit.Namespaces.Add(theNamespace);

CodeDom使用心得

2.CodeNamespace类

用来表明一个命名空间对象,这个命名空间里可以引入其他命名空间,也可以加入类

如:加入一个类到命名空间里

CodeDom使用心得

CodeTypeDeclaration ETClass  =   new  CodeTypeDeclaration( " ETMyClasses " );

CodeDom使用心得

theNamespace.Types.Add(ETClass);

CodeDom使用心得

引入一个命名空间

CodeDom使用心得

CodeNamespaceImport SystemImport  =   new  CodeNamespaceImport( " System " );

CodeDom使用心得

theNamespace.Imports.Add(SystemImport);

3.CodeTypeDeclaration类

用来表示类、结构、接口或枚举的类型声明。

声明一个类如

CodeTypeDeclaration ETClass = new CodeTypeDeclaration("ETMyClasses");

BaseTypes 基类或接口的集合,通过Add方法添加
TypeAttributes 类或接口的修饰符,如public、sealed,直接赋值,可以通过|进行或运算
Members 类成员的集合,通过Add方法可以添加字段、属性、方法等
IsEnum 设置或获取这个类型是否是枚举
 IsClass  设置或获取这个类型是否是类
 IsInterface  设置或获取这个类型是否是接口
 IsPartial  设置或获取这个类是完整声明还是部分声明
IsStruct 设置或获取这个类型是否是结构
Name 设置或获取这个类型的名称

CodeTypeDeclaration的构造函数的参数可以是这个类的名称,也可以没有参数,在实例化之后直接指定他的名称。给一个类型添加基类时可以直接使用类名称,如:ETClass.BaseTypes.Add("EntitisGateway");也可以添加CodeTypeReference的实例,如:ETClass.BaseTypes.Add(new CodeTypeReference(typeof(System.Console)));

4.CodeNamespaceImport类

用来表示映入的Namespace对象,构造函数参数为无参或命名空间名,通过CodeTypeDeclaration的Imports集合的Add方法添加命名空间的5.CodeMemberField类

用来表明类里面的字段对象,构造函数的参数为类型和名称如

CodeMemberField tableNameFiled = new CodeMemberField(typeof(string), "C_STR_TABLENAME");

这是表明定义了一个类型为string,名字为C_STR_TABLENAME的类成员字段

 Attributes 设置类的描述符号如Const、public,可以用|进行或运算 
 InitExpression  初始化字段的表达式

注意对InitExpression赋值需要使用CodePrimitiveExpression类,他是用来表示使用指定对象来初始化,如:

CodeDom使用心得

tableFieldField.InitExpression  =   new  CodePrimitiveExpression(field.FieldName);

6.CodeMemberProperty类

用来表明类里面的属性对象构造函数为无参构造函数

Name 属性的名称
Type 属性的类型,使用CodeTypeReference来进行赋值
Attributes 表明属性的修饰符public、override
HasGet 获取或设置该属性是否有get方法访问器
HasSet 获取或设置该属性是否有set方法访问器
GetStatements 表明get方法访问器的语句集合,通过Add方法添加语句
SetStatements 表明set方法访问器的语句集合,通过Add方法添加语句

给属性的get或set方法访问器添加语句如下例:

CodeDom使用心得

// 定义一个表达式,他用来表示一个局部变量的引用

CodeDom使用心得

CodeVariableReferenceExpression fieldExpression  =   new  CodeVariableReferenceExpression( " C_STR_TABLENAME " );

CodeDom使用心得

// 定一个返回语句,用来表示get的返回语句,他返回的变量是fieldExpression

CodeDom使用心得

CodeMethodReturnStatement propertyReturn  =   new  CodeMethodReturnStatement(fieldExpression);

CodeDom使用心得

// 将返回语句添加到Get的语句集合中

CodeDom使用心得

tableNameProperty.GetStatements.Add(propertyReturn);

CodeDom使用心得

7.CodeMemberMethod类

表示类里的一个成员方法,只有无参构造函数

Name 表示方法名
Attributes 方法的描述符,可以通过|进行或运算
Parameters 方法的参数集合,通过Add方法添加参数
Statements 方法中的语句集合,通过Add方法添加
CodeDom使用心得

// 构造一个方法对象

CodeDom使用心得

CodeMemberMethod LoadDataByProcessIDMethod  =   new  CodeMemberMethod();

CodeDom使用心得

// 指定方法名

CodeDom使用心得

LoadDataByProcessIDMethod.Name  =   " LoadDataByProcessID " ;

CodeDom使用心得

// 指定方法的修饰符

CodeDom使用心得

LoadDataByProcessIDMethod.Attributes  =  MemberAttributes.Public  |  MemberAttributes.Final;

CodeDom使用心得

// 给方法添加一个类型为int,名称为ProcessID的参数,通过CodeParameterDeclarationExpression来表示一个参数

CodeDom使用心得

LoadDataByProcessIDMethod.Parameters.Add( new  CodeParameterDeclarationExpression( typeof ( int ),  " ProcessID " ));

CodeDom使用心得

LoadDataByProcessIDMethod.Parameters.Add( new  CodeParameterDeclarationExpression( typeof (DbTransaction),  " objTrans " ));

CodeDom使用心得
CodeDom使用心得

// 给定方法的引用,表明要调用一个方法

CodeDom使用心得

CodeMethodReferenceExpression LoadDataExpression  =   new  CodeMethodReferenceExpression();

CodeDom使用心得

// 调用的方法名

CodeDom使用心得

LoadDataExpression.MethodName  =   " LoadDataBy " ;

CodeDom使用心得
CodeDom使用心得

// 定义方法调用的表达式,并将调用方法的引用传入,使用CodeVariableReferenceExpression来表示调用方法时传入的引用

CodeDom使用心得

CodeMethodInvokeExpression LoadDataInvoke  =   new  CodeMethodInvokeExpression(LoadDataExpression,  new  CodeVariableReferenceExpression( " C_STR_PROCESSID + "=" + ProcessID " ),  new  CodeVariableReferenceExpression( " objTrans " ));

CodeDom使用心得

// 将方法调用的表达式添加到方法额语句集合中

CodeDom使用心得

LoadDataByProcessIDMethod.Statements.Add(LoadDataInvoke);

CodeDom使用心得

// 将方法添加到类中

CodeDom使用心得

ETClass.Members.Add(LoadDataByProcessIDMethod);

CodeParameterDeclarationExpression是用指定的类型和参数名来初始化一个方法的参数。CodeVariableReferenceExpression用来表明一个局部变量名的引用

8.CodeConstructor类

用来表示一个类的构造函数,BaseConstructorArgs是用来表示基类的构造函数的参数集合,通过Add方法添加,如果基类的构造函数没有参数可以添加一个CodeVariableReferenceExpression对象,不用任何便两名初始化他,如:

CodeDom使用心得

wfPropertyConstructor1.BaseConstructorArgs.Add( new  CodeVariableReferenceExpression());

9.CodeTypeOfExpression类

用来表示typeof表达式

10.CodeThisReferenceExpression类

用来表示this的表达式

11.CodeBaseReferenceExpression类

用来表示base的表达式

CodeThisReferenceExpression和CodeBaseReferenceExpression可以和其他表达式组合在一起形成一个表达式如:

CodeDom使用心得

CodeMethodInvokeExpression InitByEntityIDInvoke  =   new  CodeMethodInvokeExpression( new  CodeBaseReferenceExpression(),  " ConstructByID " ,  new  CodeVariableReferenceExpression(m_Model.EntitiesTable  +   " ID " ),  new  CodeVariableReferenceExpression( " objTrans " ));

上面这段代码就是初始化一个方法调用的表达式,使用CodeBaseReferenceExpression和ConstructByID方法,加上两个局部变量的作为一个调用方法的参数组合成一个函数调用的表达式

12.CodeSnippetStatement类

表明一条语句,由于CodeDom为了兼容VB的语法,有些语法不支持,没有类可以调用,实在没有办法了可以通过CodeSnippetStatement来写一条语句,但这个语句不能保证在VB.Net里被通过

CodeDom使用心得

CodeSnippetStatement ExceptionStatement  =   new  CodeSnippetStatement( " throw new Exception("Property not Implementation") " );

上面这条抛出异常的语句是直接写的代码,当然CodeDom存在异常语句的对象,我这里偷了一下懒

13.IndentedTextWriter类

表明生成出来代码语句的缩进

CodeDom使用心得

 IndentedTextWriter tw  =   new  IndentedTextWriter( new  StreamWriter(sourceCodePaht  +   " / "   +  m_Model.EntitiesAssembly  +   " .cs " ,  false ),  "     " );

CodeDom使用心得

false后面的双引号之间的四个空格表明代码的缩进值为四个空格

那么我们剩下的就是生成代码了

CodeDom使用心得

IndentedTextWriter tw  =   new  IndentedTextWriter( new  StreamWriter(sourceCodePaht  +   " / "   +  m_Model.EntitiesAssembly  +   " .cs " ,  false ),  "     " );

CodeDom使用心得

CodeDomProvider provide  =   new  CSharpCodeProvider();

CodeDom使用心得
CodeDom使用心得

provide.GenerateCodeFromCompileUnit(unit, tw,  new  CodeGeneratorOptions());

CodeDom使用心得

tw.Close();

provider提供几种生成,GenerateCodeFromCompileUnit可以生成的直接生成一个代码容器里的所有代码,如果我们只写了一个类,可以通过GenerateCodeFromType方法生成,也可以生成一个命名空间里的代码,甚至可以只生成一条语句或一个表达式GenerateCodeFromNamespace、GenerateCodeFromStatement

剩下的就是要编译源代码了

CodeDom使用心得

CompilerParameters compilerParams  =   new  CompilerParameters();

CodeDom使用心得

// 指定编译选项,是类库

CodeDom使用心得

compilerParams.CompilerOptions  =   " /target:library /optimize " ;

CodeDom使用心得

// 不生成执行文件

CodeDom使用心得

compilerParams.GenerateExecutable  =   false ;

CodeDom使用心得

// 在内存中生成输出

CodeDom使用心得

compilerParams.GenerateInMemory  =   true ;

CodeDom使用心得

// 不包含调试信息

CodeDom使用心得

compilerParams.IncludeDebugInformation  =   false ;

CodeDom使用心得

// 添加编译时使用的dll,如果不是系统自带的dll,必须指定完整的路径

CodeDom使用心得

compilerParams.ReferencedAssemblies.Add( " mscorlib.dll " );

CodeDom使用心得

compilerParams.ReferencedAssemblies.Add( " System.dll " );

CodeDom使用心得

compilerParams.ReferencedAssemblies.Add( " System.Data.dll " );

CodeDom使用心得

compilerParams.ReferencedAssemblies.Add(AppDomain.CurrentDomain.SetupInformation.ApplicationBase  +   " DawnPro.WebArchitechture.DAL.dll " );

CodeDom使用心得

            compilerParams.ReferencedAssemblies.Add(AppDomain.CurrentDomain.SetupInformation.ApplicationBase  +   " DawnPro.WorkflowInterface.dll " );

CodeDom使用心得

// 指定编译后的文件的输出路径

CodeDom使用心得

compilerParams.OutputAssembly  =  outputAsseblyPath  +   " /bin/ "   +  m_Model.EntitiesAssembly  +   " .dll " ; 最后使用Provider提供的CompileAssemblyFromDom方法进行编译,provider提供了三种方法,一种是从Dom,还有一种从文件,一种从字符串中编译

CodeDom使用心得

CompilerResults result  =  provide.CompileAssemblyFromDom(compilerParams, unit); CompilerResults是这三种方法的返回值,可以通过CompilerResults的Errors集合来查看编译的错误。