做企业管理软件很难避免与Microsoft Excel打交道,常常是软件做好了,客户要求说再做一个Excel导入功能。导入Excel数据的功能的难度不大,从Excel列数据栏位的取值,验证值,再导入到数据库表中。然而一直是在做重复工作,写过不计其数的Excel导入程序,每次只是满足于问题解决,后来终于找到一个方法,实现通用的Excel数据导入。

设计通用的Excel导入功能,第一个实现要求是不能依赖Excel,客户的电脑或服务器很有可能没有安装Excel,所以微软的Office Interop一概不考虑。第二个实现要求是需要高度抽象化,也就是不依赖于具体的数据库表,这样实现了从具体表导入到通用表导入的抽象,可重复用的程度高。

第一步是生成Excel模板文件,先看字段选择界面,传入一个数据库表,可枚举表的字段,供生成Excel模板文件:

这个功能的作用是生成Excel文件,供用户输入数据。因为表名是由不同的功能窗体传递过来,实现了通用化的第一步。

生成的Excel文件,再加上我们要填写的数据,参考下面的表格。

Department.Dept Department.Description
R&D 开发部
CAM 编程
CNC 计算机锣
EDM 火花机
WC 线切割
SG 磨床
OM 其它加工
PO 打光
QC 质量控制
ASSM 模具装配
INJE 注塑部
AM 行政及管理

看Excel的表头,它包含字段定义,字段前面有加表,这样可支持主从表导入。

第二步是在上面的步骤生成的Excel文件中输入数据,再到这个界面中点击Import即可完成数据导入。

先来看一下,如何调用这个通用的导入界面功能:

   protected override void SetupImportTemplate(EntityImportArgument argument){base.SetupImportTemplate(argument);List<string> columnsList;// EntityManagerargument.EntityManager = this._departmentManager;// EntityName            argument.RootEntity = EntityType.DepartmentEntity;#region HiddenColumns#endregion#region Required Columns, columns must be selected as export columns.#region ItemcolumnsList = new List<string>();columnsList.Add(DepartmentFields.Dept.Name);columnsList.Add(DepartmentFields.Description.Name);argument.RequiredColumns.Add(EntityType.DepartmentEntity, columnsList);#endregion#endregion}

先设计好上面要调用接口,代码中解释了以下几个重要的方法:

1  要导入表  ORM映射的好处是根据实体可以找到它映射的表,根据表也可以找到它映射的实体。要跑数据验证,必须通过实体的验证类来实现,这样节省了很多验证代码。

2  保存表的方法 Manager类实现了把Excel数据保存到数据库中,对于这样通用的结构,保存方法也必须要求方法名称高度一致,比如表名是SalesOrder,它映射的实体名称是SalesOrderEntity,则对应的保存方法一定是SaveSalesOrder,这是调用时的契约,由系统强制约定。

3 值验证 如果数据库没有做强制要求输入(可空null),但是逻辑上要求一定要输入值,则需要跑实体验证。

 

数据导入使用的的第三方类库是Infragistics Excel,首先打开Excel文件,读取第一行字段名和相应的值。

using Infragistics.Documents.Excel;// Load workbook with data
Workbook workbook = Workbook.Load("department.xls");
Worksheet worksheet = workbook1.Worksheets["Sheet1"];string columnName=worksheet.Rows[0].Cells[0].Value;
 
 

构造一个内存DataSet,根据第一列的字段定义,构造表结构,例如上面的Excel文件表格,我们可以构造如下表

Department(Dept,Description)

从Excel文件中我们只能获取字段名称信息,还需要连接到数据库中,获取字段的类型信息。

SELECT * FROM sys.columns WHERE object_id=OBJECT_ID('Department')

根据这个查询,完善前面的表定义结构,类似于这样。

Department(Dept nvarchar(8),Description nvarchar(40))

这个时候就可以做基本的数据类型验证了,比如表字段的类型是数字,但Excel中的值是字符串,可抛出异常。

通过了基本的类型验证之后,我们还需要做逻辑验证,比如数据库中已经定义了R&D部门,第二次执行又插入一笔R&D的部门编码时,需要及时抛出异常,这种业务逻辑上的验证,借助于ORM框架的功能实现。

LLBL Gen Pro提供的实体类型定义,都匹配有一个验证类型,当发生值改变前,数据保存前或是删除前都可以跑验证,我们将这种复杂的验证逻辑通过实体调用来完成。

根据前面Department表的定义 ,查找系统元数据可知道它映射的实体是DepartmentEntity,我们根据数据表的值记录,构造DepartmentEntity,上面表格中的Excel数据有多行记录,则构造一个List<DepartmentEntity>,借助于反射,把内存数据库DataTale中的字段值(DataRow行)转化为实体(Entity对象)。网上有很多关于DataTable与List<Entity>转化的例子代码。

我们在保存Department方法前,主动调用Department的验证类型:

DepartmentValidator validator = new DepartmentValidator();
validator.ValidateRequiredFields(department);

最后,调用EntityManager接口中的Save方法即可:

ReflectionHelper.InvokeMethod(entityManager,”SaveDepartment”, typeof(DepartmentEntity), _department);
 

这个过程中,所操作的数据库对象是通过接口完成,实现了可扩展性,实际应用中还可导入主从表数据,需要加关联行关联Excel的每行数据之间的关系。

目前还没有实现导入三层表数据。

 

转载于:https://www.cnblogs.com/JamesLi2015/p/4700905.html

解析大型.NET ERP系统 设计通用Microsoft Excel导入功能相关推荐

  1. 解析大型.NET ERP系统 设计异常处理模块

    异常处理模块是大型系统必备的一个组件,精心设计的异常处理模块可提高系统的健壮性.下面从我理解的角度,谈谈异常处理的方方面面.我的设计仅仅限定于Windows Forms,供参考. 1 定义异常类型 . ...

  2. 解析大型.NET ERP系统:十三种界面设计模式

    成熟的ERP系统的界面应该都是从模板中拷贝出来的,各类功能的界面有规律可遵循.软件界面设计模式化或是艺术性的创作,我认可前者,模式化的界面客户容易举一反三,降低学习门槛.除了一些小部分的功能界面设计特 ...

  3. 解析大型.NET ERP系统架构设计 Framework+ Application 设计模式

    我对大型系统的理解,从数量上面来讲,源代码超过百万行以上,系统有超过300个以上的功能,从质量上来讲系统应该具备良好的可扩展性和可维护性,系统中的功能紧密关联.除去业务上的复杂性,如何设计这样的一个协 ...

  4. 解析大型.NET ERP系统 20条数据库设计规范

    数据库设计规范是个技术含量相对低的话题,只需要对标准和规范的坚持即可做到.当系统越来越庞大,严格控制数据库的设计人员,并且有一份规范书供执行参考.在程序框架中,也有一份强制性的约定,当不遵守规范时报错 ...

  5. 解析大型.NET ERP系统 高质量.NET代码设计模式

    1 缓存 Cache 系统中大量的用到缓存设计模式,对系统登入之后不变的数据进行缓存,不从数据库中直接读取.耗费一些内存,相比从SQL Server中再次读取数据要划算得多.缓存的基本设计模式参考下面 ...

  6. 解析大型.NET ERP系统 多国语言实现

    实现多国语言有许多种实现方案,无外乎是一种字符串替换技术,将界面控件的文本标签替换成相应语言的文字..NET Windows Forms实现多国语言的方法有以下几种: 1 .NET的方案,使用资源文件 ...

  7. ERP系统也有生产管理的功能,它与MES管理系统有什么差异

    企业资源管理ERP系统和MES生产执行系统都是制造企业信息管理的重要组成部分.两款系统有些功能是比较相似,但是在信息管理的方向上存在很大差异,两者的不同主要体现在六个方面. 1.管理目标 ERP系统的 ...

  8. 将 Microsoft Excel 导入至 MySQL

    将 Microsoft Excel 导入至 MySQL 一.前言 我得到了一份 Microsoft Excel 表格,里面记录了数据信息需要导入至 SQL 数据库. 如果只是导入数据,当然用 MSSQ ...

  9. import excel to mysql_将 Microsoft Excel 导入至 MySQL

    将 Microsoft Excel 导入至 MySQL 一.前言 我得到了一份 Microsoft Excel 表格,里面记录了数据信息需要导入至 SQL 数据库. 如果只是导入数据,当然用 MSSQ ...

最新文章

  1. Sketchup插件Vray户外场景设计渲染教程 Vray Next For Sketchup Exterior
  2. cmd指令大全指令_干货 | Linux 超全实用指令大全
  3. bzoj 1864: [Zjoi2006]三色二叉树
  4. ModuleNotFoundError: No module named 'win32api'
  5. Spring Boot错误errMsg: request:ok
  6. 常识分析 | 为什么 CPU 访问硬盘很慢
  7. Java中如何防止内存泄漏的发生
  8. android得到assets下面的资源
  9. python全栈测试题(一)
  10. mysql 全库查询关键字_数据库查询语句关键字总结
  11. 64位计算机很慢,win7 64位旗舰版电脑网速太慢怎么解决
  12. taptap评论爬虫
  13. apollo简易高精度地图制作
  14. 2007全球前50佳网站
  15. html鼠标经过图片有浮起效果,CSS实现鼠标滑过卡片上浮效果的示例
  16. C语言将一个数插入到已排好序的数组中
  17. 大学生课程设计 ------ Java Web课程设计(学生成绩管理系统03)
  18. OpenJudge 2986 拼点游戏
  19. js解析json js获取json里面的某个节点的数据 js解析json数据
  20. APPLE Bonjour服务导致公司网络核心Cisco 6509崩溃的案例

热门文章

  1. 无线传感器网络标准化进展与协议分析
  2. 关键词搜索商品API
  3. 用javascript调用搜狗地图实现景点的显示
  4. 传奇手游三职业1.80合击服务端三端互通版搭建教程
  5. Flex布局,真香!
  6. ORB-SLAM学习--词袋模型解析
  7. 和领导提了离职,结果他马上给我涨薪,还承诺明年晋升,该不该留下?
  8. 前端工程师所不知道的meta标签用法
  9. python海龟绘图教程自学网_python海龟绘图实例教程
  10. VueX里的mapGetters