一个老客户提出这样的需求,希望将SQLServer中的某个表的数据快速复制到SQLite数据库里面以便进行定期的备份处理,数据表的记录大概有50多万条记录,表有100个字段左右,除了希望能够快速做好外,效率是第一位的,他自己测试总是在一两个小时的时间以上。客户提出这样的需求,我我觉得肯定是没有很好的利用事务的特性,否则速度应该会快得多,但是具体能快到什么程度,心里也不太确定。于是按照这个要求,把这样大的表数据复制作为一个案例来进行研究,最终大数据的复制处理,不到20分钟就可以完成全部数据的复制更新处理。本文主要介绍这个需求如何结合实际开发的需要进行处理,达到快速高效的复制数据的目的,并提供相关的实现思路和代码供参考学习。

1、复制数据的需求及开发思路

由于客户是需要做定期的数据备份,因此这样的复制是进行的,因此大数据的复制效率肯定是很重要的,应该尽可能的短时间完成。数据表的记录大概有50多万条记录,表有100个字段左右的需要也是比常规的表数据会多一些,因此需要做好很好的测试,我们根据这样的需求背景,使用一个测试案例来对性能进行测试。

这样多字段的表,数据字段的一一对应,手工肯定是很累的,所以我们使用代码生成工具Database2Sharp来进行快速开发,这样底层的处理我们就可以不用太过关注,而且可以为不同的数据处理,生成不同的数据访问层即可。

在底层我们主要是采用了微软的Enterprise Library的数据库访问模块,因此它能够很好抽象各种数据库的事务,以适应各种不同数据库的事务处理。使用微软的Enterprise Library模块,可以很好支持SQLSever、Oracle、Mysql、Access、SQLite等数据库。

开发框架,常见的分层模式,可以分为UI层、BLL层、DAL层、IDAL层、Entity层、公用类库层等等

框架的基类我们封装了大量的通用性处理函数,包括数据访问层、业务逻辑层的基类,所有的基类函数基本上都带有一个DbTransaction trans = null 的定义,就是我们可以采用事务,也可以默认不采用事务,是一个可选性的事务参数。

如数据访问接口和基于SQLServer的数据访问类的实现图示如下所示。

在最高级的抽象基类AbstractBaseDAL的数据访问层里面,都有大量关于数据操作和相关事务的接口可以使用,因此我们在底层继承的子类,如果我们处理数据的增删改查等操作,基本上就不需要做任何扩展性代码了,这样很符合我们快速开发的目的。

在框架的整个数据访问层,我们都定义了很多公用的、带有事务参数的接口,如果我们在常规的数据处理里面,使用事务的话,那么也是很方便的事情。使用事务的批量处理,对于SQLite的操作来说,效率是非常明显的,具体可以在我之前的随笔里《使用事务操作SQLite数据批量插入,提高数据批量写入速度,源码讲解》可以了解到,他们之间的处理效率是很大差距的。

2、使用代码生成工具生成所需的代码

上面讲到,开发这样的数据复制处理程序,这样多字段的表,数据字段的一一对应,手工肯定是很累的,所以我们使用代码生成工具Database2Sharp来进行快速开发。

因此使用代码生成工具来快速生成所需要的代码,展开数据库后,从数据库节点上,右键选择【代码生成】【Enterprise Library代码生成】就可以生成标准的界面层一下的代码了,由于我们整个案例是非标准的数据复制处理,界面部分不需要利用代码生成工具进行Winform界面的生成的。

生成代码的一步步操作,最后确认一下就可以生成相关的底层代码了

最后我们生成这样的BLL、DAL、IDAL、Entity几个层的项目代码,整个项目的代码各种继承关系已经处理好了,也就具有了基类拥有的增删改查等基础操作了。

我们做两个不同数据库的复制处理操作,关键还是要生成两个不同数据库访问类的代码(也就是生成一个标准的SQLServer后,复制一份代码,修改下继承基类即可实现),如下代码是两个数据访问类的代码,不用增加任何接口即可满足当前项目的需要的了。

最终我们的项目结构如下所示。

3、进行数据复制处理的Winform界面代码逻辑

为了方便整个复制过程的进度展示(很重要),我们设计了进度条以及文字内容,展示处理过程的进度和耗时等信息,最终界面设计如下所示。

整个界面设计利用后台线程的方式对数据复制进行处理,方便及时在界面显示进度而不阻塞界面线程。

具体的界面代码如下所示。

    public partial class FrmMain : Form{private TimeSpan ExecuteTime;private int currentCount = 0;private BackgroundWorker work = new BackgroundWorker();//使用后台线程进行处理,不阻塞界面显示public FrmMain(){InitializeComponent();//定义后台线程的处理work.DoWork += work_DoWork;work.WorkerReportsProgress = true;work.ProgressChanged += work_ProgressChanged;work.RunWorkerCompleted += work_RunWorkerCompleted;}//线程完成后通知结束void work_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e){this.toolStripProgressBar1.Value = 100;this.toolStripProgressBar1.Visible = false;MessageUtil.ShowTips("操作完成");ShowMessage(this.toolStripProgressBar1.Value);//完成
        }/// <summary>/// 在界面显示文本信息/// </summary>/// <param name="percent">完成百分比</param>private void ShowMessage(int percent){if (this.ExecuteTime != null){this.lblTips.Text = string.Format("[当前完成数量:{0},完成百分比:{1}, 执行耗时:{2}毫秒 | {3}分钟{4}秒]",this.currentCount, percent, this.ExecuteTime.TotalMilliseconds, this.ExecuteTime.Minutes, this.ExecuteTime.Seconds);}}/// <summary>/// 报告进度的时候,显示相关的数量、耗时等内容/// </summary>void work_ProgressChanged(object sender, ProgressChangedEventArgs e){this.toolStripProgressBar1.Value = e.ProgressPercentage;this.statusStrip1.Refresh();ShowMessage(e.ProgressPercentage);}/// <summary>/// 后台线程执行的逻辑代码/// </summary>void work_DoWork(object sender, DoWorkEventArgs e){CopyDataUtil util = new CopyDataUtil();//使用一个Action的Lamda表达式,执行通知界面处理util.Start((percent, ts, current) =>{work.ReportProgress(percent);this.ExecuteTime = ts;this.currentCount = current;});}private void btnCopy_Click(object sender, EventArgs e){if(!work.IsBusy){//如果每次要求使用空白数据库测试,那么先删除旧数据库,再复制备份过去即可string dbfile = Path.Combine(Environment.CurrentDirectory, "localdb.db");string bakfile = Path.Combine(Environment.CurrentDirectory, "db.db");if (this.chkCopyEmptyDb.Checked && File.Exists(dbfile)){File.Delete(dbfile);File.Copy(bakfile, dbfile, true);}//显示进度条,并异步执行线程this.toolStripProgressBar1.Visible = true;work.RunWorkerAsync();}}private void FrmMain_FormClosing(object sender, FormClosingEventArgs e){//取消注册的相关事件,防止退出的时候出现异常if(work != null && work.IsBusy){work.ProgressChanged -= work_ProgressChanged; //取消通知事件work.RunWorkerCompleted -= work_RunWorkerCompleted;//取消完成事件
                work.Dispose();}}}

在上面的窗体界面代码里面,最为关键的代码就是具体后台进程的处理逻辑,如下代码所示。

        /// <summary>/// 后台线程执行的逻辑代码/// </summary>void work_DoWork(object sender, DoWorkEventArgs e){CopyDataUtil util = new CopyDataUtil();//使用一个Action的Lamda表达式,执行通知界面处理util.Start((percent, ts, current) =>{work.ReportProgress(percent);this.ExecuteTime = ts;this.currentCount = current;});}

上面的处理逻辑为了方便,把数据的复制内容放到了一个辅助类里面,并在辅助类的Start方法里面传入了界面通知的Action处理函数,这样我们在CopyDataUtil 处理的时候就可以随时进行消息的通知了。

数据复制的Start方法定义如下所示。

        /// <summary>/// 开始执行赋值/// </summary>public void Start(Action<int, TimeSpan, int> doFunc){StartTime = DateTime.Now;//计时开始
InternalCopry(doFunc);//处理数据复制逻辑,并执行外部的函数
EndTime = DateTime.Now;//计时结束}

整个辅助类CopyDataUtil 类里面定义了两个不同数据库类型的对象,方便数据库的赋值处理操作,并且定义了开始时间,结束时间,这样可以统计总共的耗时信息,如下代码所示。

    /// <summary>/// 复制数据的处理类/// </summary>public class CopyDataUtil{//使用一个计时器,对操作记录进行计时private DateTime StartTime, EndTime;//SQLServer数据库表对象private ProductSqlServer sqlserver = null;//SQLite数据表对象private ProductSqlite sqlite = null;public CopyDataUtil(){//构建对象,并指定SQLServer的数据库配置项sqlserver = new ProductSqlServer();sqlserver.DbConfigName = "sqlserver";//构建对象,并指定SQLite的数据库配置项sqlite = new ProductSqlite();sqlite.DbConfigName = "sqlite";}

整个复制数据的逻辑,主要就是基于事务性的处理,按照分页规则,每次按照一定的数量,批量从SQLServer里面取出数据,然后插入SQLite数据库里面,使用事务可以是的SQLite的数据写入非常高效快速,具体代码如下所示。

        /// <summary>/// 大数据复制的处理逻辑/// </summary>/// <param name="doFunc">外部调用的函数</param>private void InternalCopry(Action<int, TimeSpan, int> doFunc){//设置主键,并指定分页数量大小,提高检索效率string primaryKey = "h_id";int pageSize = 1000;PagerInfo info = new PagerInfo(){PageSize = pageSize, CurrenetPageIndex =1};//根据数据的总数,取得总页数int totalPageCount = 1;int totalCount = sqlserver.GetRecordCount();if (totalCount % pageSize == 0){totalPageCount = totalCount / pageSize;}else{totalPageCount = totalCount / pageSize + 1;}totalPageCount = (totalPageCount < 1) ? 1 : totalPageCount;//利用事务进行SQLite数据写入,提高执行响应效率DbTransaction trans = sqlite.CreateTransaction();if (trans != null){//根据每页数量,依次从指定的页数取数据for (int i = 1; i <= totalPageCount; i++){info.CurrenetPageIndex = i;//设定当前的页面,并进行数据获取int j = 1;List<ProductInfo> list = sqlserver.FindWithPager("1=1", info, primaryKey, false);foreach (ProductInfo entity in list){//取得当前数量和进度百分比int current = (i - 1) * pageSize + j;int percent = GetPercent(totalCount, current);//计算程序耗时,执行外部函数进行界面通知TimeSpan ts = DateTime.Now - StartTime;doFunc(percent, ts, current);//执行通知处理//如果不存在主键记录,则写入,否则更新if (!sqlite.IsExistKey(primaryKey, entity.H_id, trans)){sqlite.Insert(entity, trans);}else{sqlite.Update(entity, entity.H_id, trans);}j++;}                    }trans.Commit();}}

至此,整个项目的代码就基本上介绍完毕了,测试整个复制过程,单表50多万的数据,100个字段左右,在开发机器上20分钟不到就复制完成,确实是很不错的成绩了,如果修改为服务器的环境专门做复制处理,肯定速度还会提高不少。

大数据高效复制的处理案例分析总结相关推荐

  1. 大数据如何改善社会治理:国外“大数据社会福祉”运动的案例分析和借鉴

    一.背景 今年,国务院印发<促进大数据发展行动纲要>明确指出大数据将成为提升政府治理能力的新途径,提出:建立"用数据说话.用数据决策.用数据管理.用数据创新"的管理机制 ...

  2. 大数据江湖之即席查询与分析(下篇)--手把手教你搭建即席查询与分析Demo

    上篇小弟分享了几个"即席查询与分析"的典型案例,引起了不少共鸣,好多小伙伴迫不及待地追问我们:说好的"手把手教你搭建即席查询与分析Demo"啥时候能出?说到就得 ...

  3. 基于大数据的银行反欺诈的分析报告

    from--http://www.cnblogs.com/yueyebigdata/p/5893454.html 基于大数据的银行反欺诈的分析报告 (备注,本人主要是整理,学习他人的博客.由于大量的资 ...

  4. 物联网大数据平台软件开发架构案例解析

    物联网大数据平台软件开发架构案例解析 有人说物联网是引领信息技术的第三次浪潮. 第一次浪潮是个人电脑的出现,开创了信息时代的第一次革命,此次浪潮成就了微软.IBM等巨头. 第二次浪潮是以信息传输为特征 ...

  5. 基于大数据的动漫影视可视化分析系统

    温馨提示:文末有 CSDN 平台官方提供的学长 Wechat / QQ 名片 :) 1. 项目简介 本动漫分析系统开发语言为Python,并进行数据清洗,数据处理,并最后利用可视化技术进行动漫数据分析 ...

  6. 大数据江湖之即席查询与分析(上篇)--即席查询与分析的前世今生

    如今,大数据领域新技术层出不穷,可谓百家争鸣,甚是红火.不乏有些玩家动辄搞出个大数据平台,可谓包罗万象,号称无所不能.小弟则以为在大数据江湖中如能修炼好独门绝技,有能拿得出手的看家本领已然实属不易.小 ...

  7. 高性能计算系统——大数据与快速数据分析对高性能分析的需求

    大数据与快速数据分析对高性能分析的需求 智能家居的设备的产生必然使下一代家居服务概念化,社交网站和知识社区的日益普及,科学实验和技术计算的激增,高度可编程以及软件定义IT基础设施(服务器.存储装置.网 ...

  8. 星河璀璨 | GBASE南大通用两项成果获评2022大数据“星河”标杆、优秀案例

    12月12日,由中国信息通信研究院.中国通信标准化协会大数据技术标准推进委员会(CCSA TC601)共同组织的第六届大数据"星河(Galaxy)"案例征集结果正式公示.GBASE ...

  9. lyuyou消费大数据_基于大数据技术的电力用户行为分析及应用现状

    &Automation 基于大数据技术的电力用户行为分析及应用现状 沈玉玲,吕燕,陈瑞峰 ( 上海电气集团股份有限公司中央研究院, 上海 200070 ) 摘 要: 电力行业是大数据技术应用的 ...

最新文章

  1. 软考-信息系统项目管理师-流程管理
  2. Detect to Track and Track to Detect
  3. xadmin与mysql数据库_django和xadmin打造后台管理系统(一)-xadmin安装及使用
  4. FTP服务器软件 虚拟目录,FTP服务器软件 虚拟目录
  5. dep指定版本 go_Go 包管理工具-dep
  6. ubuntu16.04下安装openssh-server报依赖错误的解决方法
  7. java核心面试_前100多个核心Java面试问题
  8. mysql修改密码5.7_mysql数据库5.7版修改密码详细(centos7)
  9. python除数为0报错_python 错误捕获机制分析
  10. css absolute relative 定位
  11. flowable 配置自定义表单_web工作流管理系统开发之四 自定义表单
  12. 2017阿里巴巴实习生招聘编程题
  13. Win10 关闭屏幕旋转(转向)
  14. 2010考研数学二第(13)题——导数应用题
  15. 统计考勤报表 oracle对多个列求和 sum() 函数
  16. cmd中如何运行python文件_在cmd中运行.py文件: python的操作步骤
  17. 阿里企业邮箱526 Authentication failure[0]
  18. 智能手表的机遇与挑战
  19. androbench跑分性能排查
  20. CSS Grid 布局

热门文章

  1. 概率图论PGM的D-Separation(D分离)
  2. WebService(Axis2)视频教程与QQ交流群发布
  3. OC指示符assign、atomic、nonatomic、copy、retain、strong、week的解释
  4. NAT环境无法访问云端的深层次分析
  5. Zend Framework Mail通过网易免费邮箱发送邮件
  6. 初识Quartz(三)
  7. 删除当前及子文件夹中的空目录
  8. 以太坊智能合约开发第二篇:理解以太坊相关概念
  9. 使用docker-compose进行多节点部署
  10. docker查看现有容器_如何使用Docker将现有应用程序推送到容器中