Winform开发框架之通用数据导入导出操作

简介:

做了很多Winform的项目,对于数据导入,一直也有自己的理解,由于一般的业务系统,经常性的数据导入时很正常的业务需求,因为毕竟使用Excel来操作数据也很方便,或者由于系统之间的数据交换需要,我们需要提供一个入口给客户导入所需要的数据。但是导入数据的时候,不同的业务数据对应不同的Excel文件,很难做到统一,但如果是每个业务模型,都创建一个不同的导入界面来操作Excel数据,又会觉得可能某种程度上重复劳动,增加开发及维护成本。

那么有无一种介于两者之间的方法,来实现效率的最优化,并且能够统一利用好一个导入的界面呢,在开发领域,只要能想到的,一般也能做到,由于工作的需要,在我的Winform开发框架中引入了一个通用的数据导入模块,来实现这个既是统一,又是变化的业务需求,首先我们来看看能大致的模块功能介绍图,如下所示。

然后我们再来看看实际的导入模块操作界面,如下图所示

在最底的状态栏里面,但我们保存数据的时候,会调用后台线程进行数据保存,并显示数据导入的进度状态,由于是采用后台线程处理,不会阻塞当前的界面,在多文档的Winform开发框架界面中,可以切换到其他业务界面进行其他处理,不影响整体界面操作。

既然是导入界面统一,它肯定封装了一些常规操作,同时提供一些属性或者接口给外部调用对象进行操作,这样才能实现有机的统一,我们来看看具体的实现代码是如何的。

1)定义事件处理

        public delegate bool SaveDataHandler(DataRow dr);
        public event SaveDataHandler OnDataSave;
        public event EventHandler OnRefreshData;

首先我们定义一个数据保存(单行)的处理事件,然后也定义一个数据保存后,刷新主体列表的数据刷新事件,这两个都是提供给调用者实现的逻辑。

我们在这个通用的导入数据窗体,需要的就是利用后台线程调用整个逻辑进行处理数据的导入及后续的刷新操作,如下所示。

        private BackgroundWorker worker = null;

        public delegate bool SaveDataHandler(DataRow dr);
        public event SaveDataHandler OnDataSave;
        public event EventHandler OnRefreshData;

        public FrmImportExcelData()
        {
            InitializeComponent();

            this.gridView1.OptionsBehavior.AutoPopulateColumns = true;
            worker = new BackgroundWorker();
            worker.WorkerReportsProgress = true;
            worker.ProgressChanged += new ProgressChangedEventHandler(worker_ProgressChanged);
            worker.DoWork += new DoWorkEventHandler(worker_DoWork);
            worker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(worker_RunWorkerCompleted);
        }

        void worker_ProgressChanged(object sender, ProgressChangedEventArgs e)
        {
            this.progressBar1.Value = e.ProgressPercentage;
        }

        void worker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
        {
            this.progressBar1.Visible = false;
            this.progressBar1.Value = 0;

            if (OnRefreshData != null)
            {
                OnRefreshData(null, null);
            }

            string tips = e.Result as string;
            if (!string.IsNullOrEmpty(tips))
            {
                MessageDxUtil.ShowTips(tips);
                if (tips == "操作成功")
                {
                    this.gridControl1.DataSource = null;
                }
            }
        }

2)设置显示不同的模板文件

由于导入数据操作要应用于不同的业务数据,那么他们的模板肯定也不同,因此需要提供一个接口给外部,实现模板文件的修改及打开操作。

        /// <summary>
        /// 设置导入模板标题,及文件路径
        /// </summary>
        /// <param name="title"></param>
        /// <param name="filePath"></param>
        public void SetTemplate(string title, string filePath)
        {
            this.lnkExcel.Text = title;
            this.lnkExcel.Tag = filePath;
        }

        private void lnkExcel_LinkClicked(object sender, LinkLabelLinkClickedEventArgs e)
        {
            try
            {
                string templateFile = this.lnkExcel.Tag.ToString();
                Process.Start(templateFile);
            }
            catch (Exception)
            {
                MessageDxUtil.ShowWarning("文件打开失败");
            }
        }

3)显示Excel数据

我们在数据导入的时候,最好提供一个数据的显示界面给客户,方便对导入数据的核对,这样可以提高体验效果以及对数据的核对操作,减少出错的几率。具体的实现代码如下所示。数据显示的操作,可以通过操作Excel数据库的方式进行读取,然后显示数据。(其中有些接口API来自我的共用类库,需要可以到我的随笔中了解相关的类库使用。

      private void btnBrowse_Click(object sender, EventArgs e)
        {
            string file = FileDialogHelper.OpenExcel();
            if (!string.IsNullOrEmpty(file))
            {
                this.txtFilePath.Text = file;

                ViewData();
            }
        }

        private void ViewData()
        {
            if (this.txtFilePath.Text == "")
            {
                MessageDxUtil.ShowTips("请选择指定的Excel文件");
                return;
            }
            
            try
            {
                string connectString = string.Format(connectionStringFormat, this.txtFilePath.Text);
                string firstSheet = ExcelHelper.GetExcelFirstTableName(connectString);

                myDs.Tables.Clear();
                myDs.Clear();
                this.gridControl1.DataSource = null;                
                
                OleDbConnection cnnxls = new OleDbConnection(connectString);
                OleDbDataAdapter myDa = new OleDbDataAdapter(string.Format("select * from [{0}]", firstSheet), cnnxls);
                myDa.Fill(myDs, "【导入表】");

                this.gridControl1.DataSource = myDs.Tables[0];
                this.gridView1.PopulateColumns();
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.Message);
            }
        }

4)调用者给出保存数据的逻辑

由于是通用的数据导入操作,因此公用的导入界面,只能抛出相应的事件给外部进行数据保存的逻辑处理,数据导入页面只需要负责总体逻辑,具体的保存逻辑交给调用者实现,这样各司其职,共同把事情做好。下面是调用者(药品信息显示窗体中),对数据导入的操作逻辑实现。我们可以看到,它需要指定模板文件、数据刷新操作、数据保存操作,其他的交给通用数据导入界面进行处理即可。

       private string moduleName = "药品目录";
        private void btnImport_Click(object sender, EventArgs e)
        {
            string templateFile = string.Format("{0}-模板.xls", moduleName);
            FrmImportExcelData dlg = new FrmImportExcelData();
            dlg.SetTemplate(templateFile, System.IO.Path.Combine(Application.StartupPath, templateFile));
            dlg.OnDataSave += new FrmImportExcelData.SaveDataHandler(ExcelData_OnDataSave);
            dlg.OnRefreshData += new EventHandler(ExcelData_OnRefreshData);
            dlg.ShowDialog();
        }

        void ExcelData_OnRefreshData(object sender, EventArgs e)
        {
            BindData();
        }

        bool ExcelData_OnDataSave(DataRow dr)
        {
            bool success = false;
            DrugDetailInfo info = new DrugDetailInfo();
            info.DrugNo = dr["药品编码"].ToString();
            info.DrugName = dr["药品名称"].ToString();
            info.Manufacture = dr["制造商"].ToString();
            info.Formulations = dr["剂型"].ToString();
            info.Specification = dr["规格"].ToString();
            info.Unit = dr["药品单位"].ToString();
            info.Note = dr["备注信息"].ToString();
            info.StockQuantity = ConvertHelper.ToInt32(dr["库存量"].ToString(), 0);

            info.EditTime = DateTime.Now;
            info.Editor = Portal.gc.LoginInfo.Name;
            info.Dept_ID = Portal.gc.LoginInfo.Dept_ID;
            success = BLLFactory<DrugDetail>.Instance.Insert(info);
            return success;
        }

到这里,通用数据导入的操作基本上就结束了,我的处理方式是否和你的想法吻合呢,或者有更好的实现方式?

不过大家的总体思想,肯定是殊途同归,抽象封装统一的部分,并提供个性化的逻辑给外部进行处理,这样就可以实现综合的统一,提高整体的使用效率,较少今后维护的成本。

在这里顺便说一下,数据导出的操作,因为既然有导入,应该也有导出,所以我们也需要实现。它的操作代码不是很复杂,只需要把数据按照导入模板约定的字段名称导出即可,记得要和导入模板一致。

       private void btnExport_Click(object sender, EventArgs e)
        {
            string file = FileDialogHelper.SaveExcel(string.Format("{0}.xls", moduleName));
            if (!string.IsNullOrEmpty(file))
            {
                List<DrugDetailInfo> list = BLLFactory<DrugDetail>.Instance.GetAll();
                DataTable dtNew = DataTableHelper.CreateTable("序号|int,药品编码,药品名称,制造商,剂型,规格,药品单位,备注信息,库存量");
                DataRow dr;
                for (int i = 0; i < list.Count; i++)
                {
                    dr = dtNew.NewRow();
                    dr["序号"] = i + 1;
                    dr["药品编码"] = list[i].DrugNo;
                    dr["药品名称"] = list[i].DrugName;
                    dr["制造商"] = list[i].Manufacture;
                    dr["剂型"] = list[i].Formulations;
                    dr["规格"] = list[i].Specification;
                    dr["药品单位"] = list[i].Unit;
                    dr["备注信息"] = list[i].Note;
                    dr["库存量"] = list[i].StockQuantity;
                    dtNew.Rows.Add(dr);
                }

                try
                {
                    string error = "";
                    AsposeExcelTools.DataTableToExcel2(dtNew, file, out error);
                    if (!string.IsNullOrEmpty(error))
                    {
                        MessageDxUtil.ShowError(string.Format("导出Excel出现错误:{0}", error));
                    }
                    else
                    {
                        if (MessageDxUtil.ShowYesNoAndTips("导出成功,是否打开文件?") == System.Windows.Forms.DialogResult.Yes)
                        {
                            System.Diagnostics.Process.Start(file);
                        }
                    }
                }
                catch (Exception ex)
                {
                    LogHelper.Error(ex);
                    MessageDxUtil.ShowError(ex.Message);
                }
            }

以上就是我的通用数据导入导出操作,其实利用代码生成工具Database2Sharp,选定表后,自动生成的WInform界面中,就已经自动生成以上导入、导出Excel的功能代码了,已经极大简化了重复输入代码的可能性了,只需要把界面调整一下就基本上OK了,以上一家之言,欢迎拍砖或者共同探讨。

本文转自博客园伍华聪的博客,原文链接:Winform开发框架之通用数据导入导出操作,如需转载请自行联系原博主。



目录
相关文章
|
5月前
|
SQL 数据可视化 数据挖掘
SQL 在数据分析中简直太牛啦!从数据提取到可视化,带你领略强大数据库语言的神奇魅力!
【8月更文挑战第31天】在数据驱动时代,SQL(Structured Query Language)作为强大的数据库查询语言,在数据分析中扮演着关键角色。它不仅能够高效准确地提取所需数据,还能通过丰富的函数和操作符对数据进行清洗与转换,确保其适用于进一步分析。借助 SQL 的聚合、分组及排序功能,用户可以从多角度深入分析数据,为企业决策提供有力支持。尽管 SQL 本身不支持数据可视化,但其查询结果可轻松导出至 Excel、Python、R 等工具中进行可视化处理,帮助用户更直观地理解数据。掌握 SQL 可显著提升数据分析效率,助力挖掘数据价值。
174 0
|
SQL 关系型数据库 数据库
关系数据库SQL语言简介
一、关系数据库SQL语言简介 SQL(Structured Query Language)是关系型数据库管理系统(RDBMS)中最常用的语言。它是一种标准化的语言,用于存储、操作和查询数据。SQL语言具有以下特点: 1. 数据操作语言(DML):用于对数据库中的数据进行增、删、改等操作,包括INSERT、UPDATE、DELETE等命令。 2. 数据定义语言(DDL):用于定义数据库的结构,包括表、字段、索引、视图等,包括CREATE、ALTER、DROP等命令。 3. 数据查询语言(DQL):用于查询数据库中的数据,包括SELECT、FROM、WHERE等命令。 4. 数据控制语言(DCL)
299 0
|
存储 XML SQL
Activiti工作流框架学习笔记(一)之通用数据表详细介绍
Activiti工作流框架学习笔记(一)之通用数据表详细介绍
525 1
SQL通用语言之五项进阶操作
引入 何为进阶?当我们跳出了原有限制进行更新、自定义化更强的操作时就是进阶的开始。 本期我们就将进行几个原有限制的跳出—— join连接:跳出单个数据表的限制 SubQuery子查询:跳出单层逻辑的限制 常用函数:跳出工具的单一性限制 groupby分组 having过滤分组
SQL通用语言之五项进阶操作
|
数据挖掘 Python
数据分析三剑客【AIoT阶段一(下)】(十万字博文 保姆级讲解)—NumPy—Numpy 高级—通用函数(2)(十)
你好,感谢你能点进来本篇博客,请不要着急退出,相信我,如果你有一定的 Python 基础,想要学习 Python数据分析的三大库:numpy,pandas,matplotlib;这篇文章不会让你失望,本篇博客是 【AIoT阶段一(下)】 的内容:Python数据分析,
225 0
数据分析三剑客【AIoT阶段一(下)】(十万字博文 保姆级讲解)—NumPy—Numpy 高级—通用函数(2)(十)
|
数据挖掘 Python
数据分析三剑客【AIoT阶段一(下)】(十万字博文 保姆级讲解)—NumPy—Numpy 高级—通用函数(1)(九)
你好,感谢你能点进来本篇博客,请不要着急退出,相信我,如果你有一定的 Python 基础,想要学习 Python数据分析的三大库:numpy,pandas,matplotlib;这篇文章不会让你失望,本篇博客是 【AIoT阶段一(下)】 的内容:Python数据分析,
227 0
数据分析三剑客【AIoT阶段一(下)】(十万字博文 保姆级讲解)—NumPy—Numpy 高级—通用函数(1)(九)
|
SQL 关系型数据库 MySQL
MySQL基础教程2——通用知识
SQL语句可以单行也可以多行书写,以封号结尾。SQL语句可以通过空格/缩进来增加语句的可读性。MySQL数据库的SQL语句不区分大小写,关键字建议使用大写。 注释:单行注释:--注释内容或者#注释内容(MySQL数据库特有)。多行注释:/* 注释内容 */。
|
数据库 开发框架 数据安全/隐私保护