文档章节

简单代码生成器原理剖析(二)

 木宛城主
发布于 2015/03/02 19:38
字数 974
阅读 1
收藏 0
点赞 0
评论 0

上篇《简单代码生成器原理剖析(一)》分析了代码生成器的原理,查询数据库系统视图:INFORMATION_SCHEMA.TABLES、 INFORMATION_SCHEMA.COLUMNS  可以获得数据库中表、列的相关信息,再运用StringBuilder类的其AppendLine方法追加字符串,最后早运用File.WriteAllText方法将字符串写入文件。

第二版代码生成器在第一版的基础上扩展了以下功能:

  • 使用了部分类(partial):当使用大项目或自动生成的代码(如由 Windows 窗体设计器提供的代码)时,将一个类、结构或接口类型拆分到多个文件中的做法就很有用。 分部类型可能包含分部方法。
  •  使用可空类型:由于数据库中表中数据很有可能是NULL,可空类型使得数据从表中读取出来赋值给值类型更加兼容。
  • 增加了ToModel方法:将数据库表中一行数据封装成Model类的对象返回。
 
Model的实现:
/// <summary>
/// Model
/// </summary>
public static void CreateModel(CodeBuilderArgs args,string tableName)
{
DataTable dt = ExecuteDataTable(args.ConnectionString, "select * from INFORMATION_SCHEMA.COLUMNS where TABLE_NAME=@TABLE_NAME"
,new SqlParameter("TABLE_NAME",tableName));
StringBuilder sb = new StringBuilder();
sb.AppendLine("using System;");
sb.AppendLine("using System.Collections.Generic;");
sb.AppendLine("using System.Linq;");
sb.AppendLine("using System.Text;");
sb.AppendLine("namespace " + args .Namespace+ ".Model");
sb.AppendLine("{");
sb.AppendLine("partial class " + tableName);
sb.AppendLine("{");
foreach (DataRow row in dt.Rows)
{
string colName = Convert.ToString(row["COLUMN_NAME"]);
string colType = Convert.ToString(row["DATA_TYPE"]);

Type netType = GetTypeByDBType(colType);
string netTypeName;
if (netType.IsValueType)
{
netTypeName = netType + "?";
}
else
{
netTypeName = netType.ToString();
}
sb.AppendLine("public " +netTypeName +" "+ colName + " { get; set; }");
}
sb.AppendLine(" }");
sb.AppendLine(" }");
string modelDir = Path.Combine(args.OutputDir, "Model");
string modelFile = Path.Combine(modelDir, tableName+".cs");
Directory.CreateDirectory(modelDir);
File.WriteAllText(modelFile, sb.ToString());
}
DAL(数据访问层)的实现:
 
/// <summary>
///DAL
/// </summary>
/// <param name="args"></param>
/// <param name="tableName"></param>
public static void CreateDAL(CodeBuilderArgs args, string tableName)
{
StringBuilder sb = new StringBuilder();
DataTable dt = ExecuteDataTable(args.ConnectionString, "select * from INFORMATION_SCHEMA.COLUMNS where TABLE_NAME=@TABLE_NAME"
, new SqlParameter("TABLE_NAME", tableName));
sb.AppendLine("using System;");
sb.AppendLine("using System.Collections.Generic;");
sb.AppendLine("using System.Linq;");
sb.AppendLine("using System.Text;");
sb.AppendLine("using "+args.Namespace+".Model;");
sb.AppendLine("using System.Data.SqlClient;");
sb.AppendLine("using System.Data;");
sb.AppendLine("namespace "+args.Namespace+".DAL");
sb.AppendLine("{");
sb.AppendLine("partial class "+tableName+"DAL");
sb.AppendLine("{");
sb.AppendLine("public int AddNew("+tableName+" model)");
sb.AppendLine("{");

string[] cols = GetColsWithoutId(GetCols(dt, new List<string>())).ToArray();
var colsParameters=from e in cols select"@"+ e;
sb.AppendLine("string sql = \"insert into " + tableName + "(" + string.Join(",", cols) + ") output inserted.id values(" + string.Join(",", colsParameters.ToArray()) + ")\";");
sb.AppendLine("int id = (int)SqlHelper.ExecuteScalar(sql");

foreach (DataRow row in dt.Rows)
{
string colName = Convert.ToString(row["COLUMN_NAME"]);
if(!(colName.ToLower()=="id"))
{
sb.AppendLine(", new SqlParameter(\""+colName+"\", model."+colName+")");
}

}
sb.AppendLine(");");
sb.AppendLine("return id;");
sb.AppendLine("}");

var par=from e in cols select e+"="+"@"+e;
sb.AppendLine("public bool Update("+tableName+" model)");
sb.AppendLine("{");
sb.AppendLine("string sql = \"update "+tableName+" set "+string.Join(",",par.ToArray())+" where id=@id\";");
sb.AppendLine("int rows = SqlHelper.ExecuteNonQuery(sql");
foreach (DataRow row in dt.Rows)
{
string colName = Convert.ToString(row["COLUMN_NAME"]);
sb.AppendLine(", new SqlParameter(\"" + colName + "\", model." + colName + ")");
}
sb.AppendLine(");");
sb.AppendLine("return rows>0;");
sb.AppendLine("}");


sb.AppendLine("public bool Delete(int id)");
sb.AppendLine("{");
sb.AppendLine("int rows = SqlHelper.ExecuteNonQuery(\"delete from "+tableName+" where id=@id\",");
sb.AppendLine("new SqlParameter(\"id\", id));");
sb.AppendLine("return rows > 0;");
sb.AppendLine("}");

sb.AppendLine("private static "+tableName+" ToModel(DataRow row)");
sb.AppendLine("{");
sb.AppendLine(""+tableName+" model = new "+tableName+"();");
foreach (DataRow row in dt.Rows)
{
string colName = Convert.ToString(row["COLUMN_NAME"]);
string colType = Convert.ToString(row["DATA_TYPE"]);

Type netType = GetTypeByDBType(colType);
string netTypeName;
if (netType.IsValueType)
{
netTypeName = netType + "?";
}
else
{
netTypeName = netType.ToString();
}
sb.AppendLine("model."+colName+" = row.IsNull(\""+colName+"\") ? null : "+"("+netTypeName+")"+"row[\""+colName+"\"];");
}
sb.AppendLine("return model;");
sb.AppendLine("}");



//
sb.AppendLine("public "+tableName+" Get(int id)");
sb.AppendLine("{");
sb.AppendLine("DataTable dt = SqlHelper.ExecuteDataTable(\"select * from "+tableName+" where id=@id\",");
sb.AppendLine("new SqlParameter(\"id\", id));");
sb.AppendLine("if(dt.Rows.Count > 1)");
sb.AppendLine("{");
sb.AppendLine("throw new Exception(\"more than 1 row was found\");");
sb.AppendLine("}");
sb.AppendLine("if(dt.Rows.Count <= 0)");
sb.AppendLine("{");
sb.AppendLine("return null; ");
sb.AppendLine("}");
sb.AppendLine("DataRow row = dt.Rows[0];");
sb.AppendLine(""+tableName+" model = ToModel(row);");
sb.AppendLine("return model;");
sb.AppendLine("}");

sb.AppendLine("public IEnumerable<"+tableName+">ListAll()");
sb.AppendLine("{");
sb.AppendLine("List<"+tableName+"> list = new List<"+tableName+">();");
sb.AppendLine("DataTable dt = SqlHelper.ExecuteDataTable(\"select * from "+tableName+"\");");
sb.AppendLine("foreach (DataRow row in dt.Rows)");
sb.AppendLine("{");
sb.AppendLine("list.Add(ToModel(row));");
sb.AppendLine("}");
sb.AppendLine("return list;");
sb.AppendLine("}");
sb.AppendLine("}");
sb.AppendLine("}");

string DALlDir = Path.Combine(args.OutputDir,"DAL");
string DALFile = Path.Combine(DALlDir,tableName+"DAL.cs");
Directory.CreateDirectory(DALlDir);
File.WriteAllText(DALFile,sb.ToString());
}
private static IEnumerable<string> GetColsWithoutId(List<string> cols)
{
var colArray = from col in cols where col.ToLower() != "id" select col;
return colArray;
}
private static List<string> GetCols(DataTable dt, List<string> cols)
{
foreach (DataRow row in dt.Rows)
{
string colName = Convert.ToString(row["COLUMN_NAME"]);
cols.Add(colName);
}
return cols;
}
BLL的实现:
/// <summary>
/// BLL
/// </summary>
/// <param name="args"></param>
/// <param name="tableName"></param>
public static void CreateBLL(CodeBuilderArgs args,string tableName)
{
StringBuilder sb = new StringBuilder();
sb.AppendLine("using System;");
sb.AppendLine("using System.Collections.Generic;");
sb.AppendLine("using System.Linq;");
sb.AppendLine("using System.Text;");
sb.AppendLine("using "+args.Namespace+".Model;");
sb.AppendLine("using "+args.Namespace+".DAL;");
sb.AppendLine("namespace "+args.Namespace+".BLL");
sb.AppendLine("{");
sb.AppendLine("partial class "+tableName+"BLL");
sb.AppendLine("{");
sb.AppendLine("public int AddNew("+tableName+" model)");
sb.AppendLine("{");
sb.AppendLine("return new "+tableName+"DAL().AddNew(model);");
sb.AppendLine("}");
sb.AppendLine("public bool Delete(int id)");
sb.AppendLine("{");
sb.AppendLine("return new "+tableName+"DAL().Delete(id);");
sb.AppendLine("}");
sb.AppendLine("public bool Update("+tableName+" model)");
sb.AppendLine("{");
sb.AppendLine("return new "+tableName+"DAL().Update(model);");
sb.AppendLine("}");
sb.AppendLine("public "+tableName+" Get(int id)");
sb.AppendLine("{");
sb.AppendLine("return new "+tableName+"DAL().Get(id);");
sb.AppendLine("}");
sb.AppendLine("public IEnumerable<"+tableName+"> ListAll()");
sb.AppendLine("{");
sb.AppendLine("return new "+tableName+"DAL().ListAll();");
sb.AppendLine("}");
sb.AppendLine("}");
sb.AppendLine("}");
string BLLDir = Path.Combine(args.OutputDir,"BLL");
string BLLFile = Path.Combine(BLLDir,tableName+"BLL.cs");
Directory.CreateDirectory(BLLDir);
File.WriteAllText(BLLFile,sb.ToString());
}
详细代码可参考:

© 著作权归作者所有

共有 人打赏支持
粉丝 2
博文 222
码字总数 199010
作品 0
黄浦
如何基于yeoman快速编写脚手架工具(系列:使用篇上)

如何基于yeoman快速编写脚手架工具(系列:使用篇上) 由于关于yeoman编写脚手架的文章太少,所以推出此文,注意本文介绍的yoeman generator-generator是基于 0.8.0以前的版本,0.1.0以后的版...

刘学炜 ⋅ 2016/01/23 ⋅ 0

TensorFlow实现seq2seq

前言 前面在《深度学习的seq2seq模型》文章中已经介绍了seq2seq结构及其原理,接下去这篇文章将尝试使用TensorFlow来实现一个seq2seq网络结构,该例子能通过训练给定的训练集实现输入某个序列...

sea-boat ⋅ 2017/09/14 ⋅ 0

Python中的yield

一、迭代器(iterator) 在Python中,for循环可以用于Python中的任何类型,包括列表、元祖等等,实际上,for循环可用于任何“可迭代对象”,这其实就是迭代器 迭代器是一个实现了迭代器协议的对...

翼动动空 ⋅ 2016/05/09 ⋅ 0

Mybatis-plus 1.2.2 发布,MyBatis 增强工具包

mybatis-plus 1.2.2 发布了。MyBatis 增强工具包,简化 CURD 操作。 启动加载 XML 配置时注入单表 SQL 操作 ,为简化开发工作、提高生产率而生。 mybatis-plus 实践及架构原理 与其他工具包区...

青苗 ⋅ 2016/03/14 ⋅ 9

Javac编译原理:Item(1)——基本介绍

首先,必须明确Java语言规范和Java虚拟机规范不是一回事。 Java语言规范描述了Java语言有哪些词法和语法,而Java虚拟机也有Java虚拟机规范,它们都有自己的词法和语法解析规则,而且解析规则...

_Roger_ ⋅ 2016/02/16 ⋅ 0

详解生成器、迭代器

1.迭代     要搞清楚什么关于迭代器,生成器,可迭代对象,前提是我们要理解何为迭代。   第一,迭代需要重复进行某一操作   第二,本次迭代的要依赖上一次的结果继续往下做,如果中...

xsster ⋅ 2017/07/25 ⋅ 0

并发包中ThreadLocalRandom类原理剖析

2.2、 并发包中ThreadLocalRandom类原理剖析 ThreadLocalRandom类是JDK7在JUC包下新增的随机数生成器,它解决了Random类在多线程下多个线程竞争内部唯一的原子性种子变量而导致大量线程自旋重...

今天你不奋斗明天你就落后 ⋅ 2017/12/24 ⋅ 0

生成对抗网络(GANs)最新家谱:为你揭秘GANs的前世今生

作者:Guim Perarnau 编译:Katherine Hou、朝夕、Katrine Ren、Shan LIU、笪洁琼、钱天培 生成对抗网络(GAN)一经提出就风光无限,更是被Yann Lecun誉为“十年来机器学习领域最有趣的想法”...

dzjx2eotaa24adr ⋅ 2017/12/04 ⋅ 0

PHP关键字之01 - yield

从PHP5.5开始,可以使用生成器来处理一个序列。生成器是一个函数,它不会调用return来返回一个值,而会调用yield(可能在一个循环中调用)。有了这样一个生成器,可以在原先使用数组的地方调...

Nosee123 ⋅ 05/29 ⋅ 0

JEECG快速开发框架学习

JEECG开源代码地址:http://git.oschina.net/jeecg/jeecg JEECG开源社区地址:http://www.jeecg.org/ JEECG开发指南:http://www.jeecg.org/book/jeecgv3.html 前言: 随着WEB UI 框架(Easy......

fatwowzb ⋅ 2015/04/03 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

Android JNI 读写Bitmap的方法

Java层创建Bitmap,通过JNI将Bitmap传到C/C++进行处理 Java部分 public static native boolean greenBitmap(Bitmap bitmap); C/C++部分 JNIEXPORT jboolean JNICALL Java_com_test_Test_gree......

国仔饼 ⋅ 6分钟前 ⋅ 0

一次性让你懂async/await,解决回调地狱

什么是async? 欢迎留言讨论 async 函数是 Generator 函数的语法糖。使用 关键字 async 来表示,在函数内部使用 await 来表示异步。相较于 Generator,async 函数的改进在于下面四点: 内置执...

阿K1225 ⋅ 6分钟前 ⋅ 0

angular常用命令

.下载更新操作 1.利用npm下载angular的命令行工具AngularCLI: npm install -g @angular/cli 2.下载jquery: npm install --save jquery 3.更新npm: npm i -g npm 4.更新angular: ng update ......

消散了的诗意 ⋅ 9分钟前 ⋅ 0

window.print 页面打印

定义和用法 print() 方法用于打印当前窗口的内容。 语法 window.print(); window.print() 实际上,是浏览器打印功能菜单的一种程序调用。与点击打印功能菜单一样,不能精确分页,不能设置纸型...

初学者的优化 ⋅ 9分钟前 ⋅ 0

魔兽世界 7.0版本上 PVE装备全攻略

  T套 因为大家应该都会打穿副本的所以具体是哪个boss我就不说了。   T1: 所有套装都在【熔火之心】出   T2: 头原来是在【奥妮克希亚的巢穴】改到黑翼之巢的奈法利安了,腿是在【熔火之...

wangchen1999 ⋅ 10分钟前 ⋅ 0

java.math.BigDecimal使用小结

原文地址 java.math.BigDecimal使用小结 divide方法 使用BigDecimal.divide方法时一定要考虑: 除数是否为0 商是否是无限小数 正确的使用方式 判断除数是否为0,是0做另外的处理逻辑 调用除法...

666B ⋅ 12分钟前 ⋅ 0

关于qstring转char乱码问题。

if (OpenClipboard(NULL)) { HGLOBAL hgClip; EmptyClipboard(); QByteArray byay = FValue.toLocal8Bit(); //转latin编码 char *bochsrc_line = byay.data(); hgClip = GlobalAlloc(GMEM_DD......

backtrackx ⋅ 13分钟前 ⋅ 0

了解SSH加密和连接过程

介绍 SSH或安全shell是安全协议,也是安全管理远程服务器的最常用方式。通过使用多种加密技术,SSH提供了一种机制,用于在双方之间建立加密安全连接,对彼此进行身份验证,以及来回传递命令和...

吴伟祥 ⋅ 19分钟前 ⋅ 0

微信小程序

小程序的基础配置:导航栏和tabbar 在app.json文件中配置导航栏和tabrbar 导航栏的设置 设置导航,背景黑色,文字白色,文字内容 { "pages":[ "pages/index/index", "pages/logs/l...

上官清偌 ⋅ 22分钟前 ⋅ 0

【转】百度坐标坐标系之间的转换(JS版代码)

/** * Created by Wandergis on 2015/7/8. * 提供了百度坐标(BD09)、国测局坐标(火星坐标,GCJ02)、和WGS84坐标系之间的转换 *///定义一些常量var x_PI = 3.1415926535897932...

HAVENT ⋅ 24分钟前 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部