从Excel转Access的一个方法说开去(DataRow的state状态)
因为客户对access不太熟悉,更喜欢玩EXCEL。但是系统要求导入ACCESS。所以我们得做个把EXCEL转换成Access的小工具。(别问我为啥不让系统直接导入excel....我不知道!),然后耗费了点时间写了个公用的方法,如下:
- /// <summary>
- ///
- /// </summary>
- /// <param name="excelpath">excel路径</param>
- /// <param name="exceltablename">Excel表名</param>
- /// <param name="accessmodelpath">access模版路径</param>
- /// <param name="accesspath">access输出路径</param>
- /// <param name="accesstablename">access表名</param>
- /// <param name="msg">错误信息</param>
- /// <returns></returns>
- public static bool ExcelToAccess(string excelpath, string exceltablename, string accessmodelpath, string accesspath, string accesstablename, ref string msg)
- {
- try
- {
- DataTable dt = ExcelToDataTable(excelpath, exceltablename);
- if (dt == null)
- {
- msg = "Excel没有任何内容";
- return false;
- }
- accesspath = accesspath + "\\生成的access.mdb";
- System.IO.File.Copy(accessmodelpath, accesspath, true);
- string connStr = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + accesspath + ";User Id=admin;Password=;";
- string sql = "Select * from " + accesstablename + "";
- System.Data.OleDb.OleDbConnection conn = new System.Data.OleDb.OleDbConnection(connStr);
- System.Data.OleDb.OleDbDataAdapter da = new System.Data.OleDb.OleDbDataAdapter(sql, conn);
- System.Data.DataTable acedt = new DataTable();
- da.Fill(acedt);
- System.Data.OleDb.OleDbCommandBuilder ocb = new System.Data.OleDb.OleDbCommandBuilder(da);
- da.InsertCommand = ocb.GetInsertCommand();
- DataRow[] drs = dt.Select();
- foreach (DataRow r in drs)
- {
- acedt.ImportRow(r);
- }
- // acedt.AcceptChanges();//此处不能有
- da.Update(acedt);
- da.Dispose();
- conn.Close();
- conn.Dispose();
- return true;
- }
- catch (Exception ex)
- {
- msg = ex.Message;
- return false;
- }
- }
这里用了模版,用IO的创建代替用sql创建access,应该是个比较好点的办法,当然,前提是你得实现有个模版。
然后是 ExcelToDataTable方法的代码:
- /// <summary>
- /// Excel转换成datatable
- /// </summary>
- /// <param name="excelpath"></param>
- /// <param name="exceltablename"></param>
- /// <returns></returns>
- public static DataTable ExcelToDataTable(string excelpath, string exceltablename)
- {
- try
- {
- //
- string strCon =string.Format( "Provider=Microsoft.Jet.OLEDB.4.0;Data Source={0};Extended Properties='Excel 8.0;IMEX=1'",excelpath);
- OleDbConnection Conn = new OleDbConnection(strCon);
- string strCom = "SELECT distinct * FROM [" + exceltablename + "$] where 1=1 ";
- OleDbDataAdapter da = new OleDbDataAdapter(strCom, Conn);
- DataTable dt = new DataTable();
- da.AcceptChangesDuringFill = false;
- da.Fill(dt);
- return dt;
- }
- catch(Exception ex)
- {
- return null;
- }
- }
关键点是 da.AcceptChangesDuringFill = false;这句。最初的时候,我并没有写这一局。
一开始我没主意到这个,跑这个方法的时候,运行起来没有任何问题,但是,打开生成的access,是空数据!
查阅资料外加问人,发现了此时dt的datarow的rowstate为unchanged;
知道问题所在了,就去百度,添加该句后状态改变为added;
然后终于可以添加到access了。
现在看起来不是很难,但是,为了找到这个原因,耗费了一天的时间。
下面是我查找参考资料的时候从MSDN看到的关于DataRow.RowState的一些代码:
- static void Main(string[] args)
- {
- // Run a function to create a DataTable with one column.
- DataTable table = MakeTable();
- DataRow row;
- // Create a new DataRow.
- row = table.NewRow();
- // Detached row.
- Console.WriteLine("New Row " + row.RowState);
- table.Rows.Add(row);
- // New row.
- Console.WriteLine("AddRow " + row.RowState);
- table.AcceptChanges();
- // Unchanged row.
- Console.WriteLine("AcceptChanges " + row.RowState);
- row["FirstName"] = "Scott";
- // Modified row.
- Console.WriteLine("Modified " + row.RowState);
- row.Delete();
- // Deleted row.
- Console.WriteLine("Deleted " + row.RowState);
- Console.ReadKey();
- }
- private static DataTable MakeTable()
- {
- // Make a simple table with one column.
- DataTable table = new DataTable("table");
- DataColumn dcFirstName = new DataColumn(
- "FirstName", Type.GetType("System.String"));
- table.Columns.Add(dcFirstName);
- return table;
- }
才知道原来不同的操作,datarow的状态是不同的。
下面来说说他的状态:
DataRowState枚舉有如下几种状态:
add 该行已添加到 DataRowCollection 中,AcceptChanges 尚未调用
Deleted 该行已通过 DataRow 的 Delete 方法被删除。
Detached 该行已被创建,但不属于任何 DataRowCollection。DataRow 在以下情况下立即处于此状态:创建之后添加到集合中之前;或从集合中移除之后。
Modified 该行已被修改,AcceptChanges 尚未调用。
Unchanged 该行自上次调用 AcceptChanges 以来尚未更改。
要正確理解上面這集中狀態應該對AcceptChanges有很好的理解:
提交自上次调用 AcceptChanges 以来对该行进行的所有更改。(意思就是对以上做的更改的确认)
在调用 AcceptChanges 时,EndEdit 方法被隐式调用,以便终止任何编辑。如果行的 RowState 是“Added”或“Modified”,则 RowState 变成“Unchanged”。如果 RowState 是“Deleted”,则该行将被移除。
如何获取已删除行的信息:获取行的DataRowVersion.Original版本就可以了.
从Excel转Access的一个方法说开去(DataRow的state状态)的更多相关文章
- 从一个Bug说开去--解决问题的思路,Linked Server, Bulk Insert, DataTable 作为参数传递
声名— 部分内容为杜撰,如有雷同,不胜荣幸! 版权所有,如要引用,请标明出处! 如果打赏,请自便! 1 背景介绍 最近一周在忙一个SQL Server 的Bug,一个简单的Bug,更新两张 ...
- 从java main方法说开去(转)
刚刚接触java语言时,接触的便为一个java main方法.我们知道这样程序就可以运行了,但是程序是怎么运行起来的我们却不知道. 众所周知,当执行一个java程序时,首先会启动一个JVM虚拟机进程, ...
- PHP导出Excel一个方法轻松搞定
/** * 导出数据为excel表格 *@param $data 一个二维数组,结构如同从数据库查出来的数组 *@param $title excel的第一行标题,一个数组,如果为空则没有标题 *@p ...
- SharePoint Server 2013 Excel Web Access无法显示
环境信息:SharePoint Server 2013 中文版,版本为15.0.4420.1017 Windows Server 2008 r2中文版 Sql Server 2012 问题描述:在Sh ...
- R语言读取excel文件的3种方法
R读取excel文件中数据的方法: 电脑有一个excel文件,原始的文件路径是:E:\R workshop\mydata\biom excel数据为5乘2阶矩阵,元素为 ...
- 详细解释VB连接access几种方法数据库
在VB中,连接ACCESS数据库的方法主要有以下三种 使用ADO对象,通过编写代码訪问数据库 Connection 对象 ODBC数据源 使用ADO Data 控件高速创建数据库连接 有三种连接方法 ...
- 将DataTable内容导出到Excel表格的两种方法
方法一:循环DataTable单元格内容拼接字符串,利用StreamWriter的Write方法将字符串写入Excel文件中 这种方法很实现很简单.拼接字符串时,每个单元格之间添加'\t'(表示一个占 ...
- Microsoft Office Excel cannot access the file, There are several possible reasons
今天在做EXCEL打印读取模板时报错了,错误信息如下: Microsoft Excel cannot access the file 'D:\xx.xlsx'. There are several p ...
- 多个EXCEL文件合并成一个
Python的numpy处理起来会比较方便,有空实现一下,这里是Excel内部代码的方式: 合并方法如下: 1.需要把多个excel表都放在同一个文件夹里面,并在这个文件夹里面新建一个excel.如图 ...
随机推荐
- 10-2[RF] OOB validation
main idea: 在使用bootstrap生成gi的训练集时,会有一部分数据没有被选中,使用这一部分数据(OOB)进行validation. 1.数据没有被选中的概率 假设训练集大小为N,使用bo ...
- Spring、基本类型属性和集合类型属性的注入
Spring 还可以对基本属性和集合类型属性进行注入: public interface PersonIService { public String getBaseProperty(); publi ...
- 全国计算机等级考试二级教程-C语言程序设计_第6章_字符型数据
#include <stdio.h> main() { char c; char d; c = ; d = '; if (c == d) { printf("yes\n" ...
- 调magento自定义模板发邮件
1. 设置邮件模板 <global> <template> <email> <custom_email_template1 module="Samp ...
- 产品专家Marty Cagan:不做仅仅会编码的人
Marty Cagan是享有世界声誉的产品管理专家,曾担任Netscape副总裁.eBay产品管理及设计高级副总裁. 近日,记者在"PM-China首届产品经理高峰论坛"上对他做了 ...
- Math.round(11.5)等于()Math.round(-11.5)等于()
几天前去面试,这道简单的题目居然做错了,看来基础就是慢慢积累的.并不断使用和复习才会成为高手,假设基础不是那么熟练.恐怕在成为高手的路上会困难重重.所以在做项目的间歇时间.偶尔回顾一下最基础的知识.是 ...
- C#中使用日志类,添加dll时出现错误
警告 1 未能解析引用的程序集 “log4net, Version=1.2.10.0, Culture=neutral, PublicKeyToken=1b44e1d426115821, proces ...
- MongoDB无法启动的解决方法
http://dmyz.org/archives/423 遇到MongoDB突然无法启动,第一反应是删除mongod.lock.这个文件在MongoDB的数据库目录下,默认是/data/db.这是最常 ...
- Java Tomcat Glassfish Weblogic远程debug(remote debug)
tomcat ./catalina.sh jpda start 这条命令启动tomcat,它就会监听8000端口,等待调试器的连接. 默认监听8000端口,通过设置环境变量JPDA_ADDRESS指定 ...
- EC读书笔记系列之11:条款20、21
条款20 宁以pass-by-reference-to-const替换pass-by-value 记住: ★尽量以pass-by-reference-to-const替换pass-by-value.前 ...