技术开发 频道

使用CodeSmith加快开发进度

【IT168 技术文档】

  随着开发进度表的缩短,开发者不可能使用太多的时间。代码生成工具可生成常见的通用代码,为你节省宝贵的时间。让我们通过CodeSmith工具来了解一下这种开发趋势吧!

  设计式样为常见的编程问题提供了标准解决方法。但是,开发人员发现自己总是要不断地输入同样的代码。代码生成工具提供了起步所必须的基础代码,从而解决这一问题。它加快了开发过程,并保证了代码的稳定性与可靠性。让我们通过CodeSmith来详细了解一下代码生成过程。

  1 CodeSmith介绍

  CodeSmith是一个基于模板的代码生成器。模板是所生成代码的式样。开发者或构架师可应用CodeSmith来生成任何文本语言的代码。其结果(生成的代码)可通过属性来自定义并包含在众多的标准属性类型之中。另外,用户还可建立自定义的属性类型。

   针对.NET Framework而言,性质可以是任何拥有设计器的.NET对象。例如,它可以是一个简单的分配标题的字符串性质。另一方面,可用TableSchema对象来访问数据库表中的一切内容。

   CodeSmith工具的一个强项是它的语法,其与ASP.NET的方法相当。实际上,你可以在CodeSmith模板中应用C#、VB.NET或Jscript。事实上,CodeSmith可输入任何ASCII语言。

   CodeSmith 包括两个工具,一个是CCodeSmithStudio.exe是用来设计和编译模板;另一个是CodeSmith.exe是用来运行模板生成代码的, CodeSmith.exe还可以与VS.NET集成,成为VS.NET的一个外部工具。

  2 CodeSmith入门

   创建好一个模板后,第一步要指明这是一个C#语言的模板。

<%@ CodeTemplate Language="C#" TargetLanguage="C#" Description="Generates a class including a special informational header" %>

   第二步,我们要指明模板在生成代码时的属性,即生成代码需要的输入值变量。 

<%@ Property Name="NameSpace" Type="String" Category="Context" Description="The namespace to use for this class" %>

   如上所示,在进行代码生成时,在CodeSmith Explorer中选择模板后生成代码的窗口中,变量的名称为NameSpace,类型是String,类别是Context,当用户选中这个属性时对于属性的描述Description。

   我们可以按照C#语言的语法去使用定义的变量,例如:

/////////////////////////////////////////////////////////////////////////////////////// // File: <%=ClassName%>.cs

   例如,下面这个例子模板使用了上面介绍的知识。

<%@ CodeTemplate Language="C#" TargetLanguage="C#" Description="Generates a class including a special informational header" %> <%@ Property Name="NameSpace" Type="String" Category="Context" Description="The namespace to use for this class" %> <%@ Property Name="ClassName" Type="String" Category="Context" Description="The name of the class to generate" %> <%@ Property Name="DevelopersName" Type="String" Category="Context" Description="The name to include in the comment header" %> /////////////////////////////////////////////////////////////////////////////////////// // File: <%=ClassName%>.cs // Description: Enter summary here after generation. // --------------------- // Copyright ?<%= DateTime.Now.Year %> Our Client // --------------------- // History // <%= DateTime.Now.ToShortDateString() %> <%= DevelopersName%> Original Version /////////////////////////////////////////////////////////////////////////////////////// using System; namespace <%=NameSpace %> { /// <summary> /// Summary description for <%=ClassName %>. /// </summary> public class <%=ClassName %> { public <%=ClassName %>() { // // TODO: Add constructor logic here // } } }

 

  然后我们在CodeSmith Explorer中双击这个模板,就会看到相应的属性界面,这里的属性均是我们在前边定义的属性。

  按下Generate按钮生成,即可实现一个简单的类代码的生成。

1 /////////////////////////////////////////////////////////////////////////////////////// 2 // File: MyClass.cs 3 // Description: Enter summary here after generation. 4 // --------------------- 5 // Copyright ?2003 Our Client 6 // --------------------- 7 // History 8 // 12/2/2003 Mr. Smith Original Version 9 /////////////////////////////////////////////////////////////////////////////////////// 10 11 using System; 12 13 namespace MyNameSpace 14 { 15 /// <summary> 16 /// Summary description for MyClass. 17 /// </summary> 18 public class MyClass 19 { 20 public MyClass() 21 { 22 // 23 // TODO: Add constructor logic here 24 // 25 } 26 } 27 }

 

  生成后的代码即可放入Visual Studio .NET中使用,我们使用CodeSmith的目的就是为了快速高效的开发。

  3 CodeSmith与数据库结合

  本文将介绍CodeSmith与数据库进行交互生成相应的存储过程,本例使用的数据库为SQL Server 2000。

  在与数据库进行交互时,我们使用到了一个CodeSmith自带的组件SchemaExplorer,利用这个组件我们可以访问数据库的数据表、存储过程、视图等,并可以得到相应的数据类型、标识列、列的(字段)名称等信息。

  下面这个例子是教我们如何生成一个存储过程。

  使用的是SQL Server 2000自带的Northwind数据库,生成一个关于Orders订单表的更新存储过程。

  第一步还是指明模板使用的语言和生成的目标语言。

<%@ CodeTemplate Language="C#" TargetLanguage="T-SQL" Description="Generates a update stored procedure." %>

 

  第二步就是我们要加载使用访问数据库的组件SchemaExplorer,并声明其使用的命名空间。

<%@ Assembly Name="SchemaExplorer" %> <%@ Import Namespace="SchemaExplorer" %>

 

  因为是针对表去生成存储过程,则首先要定义一个存储表名称使用的变量,然后指明这个变量类型为数据库中的表,这样我们可以通过这个数据表类型的变量得到相应的表的信息。

<%@ Property Name="SourceTable" Type="SchemaExplorer.TableSchema" Category="Context" Description="Table that the stored procedures should be based on." %>

 

  如果想访问视图的话,则将变量类型Type中的SchemaExplorer.TableSchema修改为SchemaExplorer.ViewSchema即可。

   得到表名的方法如下:

CREATE PROCEDURE dbo.Update<%= SourceTable.Name %>

 

   下面利用循环语句遍历表的各个列,拼出存储过程需要传递的参数。

<% for (int i = 0; i < SourceTable.Columns.Count; i++) { %> <%= GetSqlParameterStatement(SourceTable.Columns[i]) %><% if (i < SourceTable.Columns.Count - 1) { %>,<% } %> <% } %>

 

   调用的GetSqlParameterStatement方法是用来生成参数的字符串,例如生成“@CustomerID nchar(5)”,后边紧跟的if判断是用来生成参数之间相隔使用的逗号的。

   生成参数字符串的方法,参数为SchemaExplorer.ColumnSchema列类型。

1 <script runat="template"> 2 public string GetSqlParameterStatement(ColumnSchema column) 3 { 4 string param = "@" + column.Name + " " + column.NativeType; 5 6 switch (column.DataType) 7 { 8 case DbType.Decimal: 9 { 10 param += "(" + column.Precision + ", " + column.Scale + ")"; 11 break; 12 } 13 default: 14 { 15 if (column.Size > 0) 16 { 17 param += "(" + column.Size + ")"; 18 } 19 break; 20 } 21 } 22 23 return param; 24 } 25 </script>

 

   下面来生成需要更新的字段,更新时仅能更新非主键字段的值,在SchemaExplorer中支持这种区别,使用SourceTable.NonPrimaryKeyColumns即可得到非主键字段的集合。

1UPDATE [<%= SourceTable.Name %>] SET 2 <% for (int i = 0; i < SourceTable.NonPrimaryKeyColumns.Count; i++) { %> 3 [<%= SourceTable.NonPrimaryKeyColumns[i].Name %>] = @<%= SourceTable.NonPrimaryKeyColumns[i].Name %><% if (i < SourceTable.NonPrimaryKeyColumns.Count - 1) { %>,<% } %> 4 <% } %>

 

   然后再使用SourceTable.PrimaryKey.MemberColumns得到数据表中的主键集合,生成更新条件:

1 WHERE 2 <% for (int i = 0; i < SourceTable.PrimaryKey.MemberColumns.Count; i++) { %> 3 <% if (i > 0) { %>AND <% } %> 4 [<%= SourceTable.PrimaryKey.MemberColumns[i].Name %>] = @<%= SourceTable.PrimaryKey.MemberColumns[i].Name %> 5 <% } %>

 

   以下为整体的代码结构:

1 <%@ CodeTemplate Language="C#" TargetLanguage="T-SQL" 2 Description="Generates a update stored procedure." %> 3 4 <%@ Property Name="SourceTable" Type="SchemaExplorer.TableSchema" 5 Category="Context" 6 Description="Table that the stored procedures should be based on." %> 7 8 <%@ Assembly Name="SchemaExplorer" %> 9 10 <%@ Import Namespace="SchemaExplorer" %> 11 12 <script runat="template"> 13 public string GetSqlParameterStatement(ColumnSchema column) 14 { 15 string param = "@" + column.Name + " " + column.NativeType; 16 17 switch (column.DataType) 18 { 19 case DbType.Decimal: 20 { 21 param += "(" + column.Precision + ", " + column.Scale + ")"; 22 break; 23 } 24 default: 25 { 26 if (column.Size > 0) 27 { 28 param += "(" + column.Size + ")"; 29 } 30 break; 31 } 32 } 33 34 return param; 35 } 36 </script> 37 38 ----------------------------------------------------------------- 39 -- Date Created: <%= DateTime.Now.ToLongDateString() %> 40 -- Created By: Generated by CodeSmith 41 ----------------------------------------------------------------- 42 43 CREATE PROCEDURE dbo.Update<%= SourceTable.Name %> 44 <% for (int i = 0; i < SourceTable.Columns.Count; i++) { %> 45 <%= GetSqlParameterStatement(SourceTable.Columns[i]) %><% if (i < SourceTable.Columns.Count - 1) { %>,<% } %> 46 <% } %> 47 AS 48 49 UPDATE [<%= SourceTable.Name %>] SET 50 <% for (int i = 0; i < SourceTable.NonPrimaryKeyColumns.Count; i++) { %> 51 [<%= SourceTable.NonPrimaryKeyColumns[i].Name %>] = @<%= SourceTable.NonPrimaryKeyColumns[i].Name %><% if (i < SourceTable.NonPrimaryKeyColumns.Count - 1) { %>,<% } %> 52 <% } %> 53 WHERE 54 <% for (int i = 0; i < SourceTable.PrimaryKey.MemberColumns.Count; i++) { %> 55 <% if (i > 0) { %>AND <% } %> 56 [<%= SourceTable.PrimaryKey.MemberColumns[i].Name %>] = @<%= SourceTable.PrimaryKey.MemberColumns[i].Name %> 57 <% } %>

 

   联系作者:tyrwjz2003@vip.sina.com

0
相关文章