如何在不使用OleDbCommandBuilder情况下使用OleDbDataAdapter更新Access数据库记录
我在博客园的博问和微软论坛都曾经请教了这个问题(问题链接),可能我的问题太简单,并没有获得太多解答。
到今天为止,我自己通过查找和摸索,基本把这个问题解决了,还是记录下来,供其他朋友参考。
第一次解决这么复杂的问题,我尽量用通俗的话语把问题说明白。
一、这个问题怎么来的?
众所周知,Access是典型的关系型数据库,数据库各表或多或少的都有对应关系。对数据库进行操作,往往就不能只操作一个表,最常见的场景就是:在使用基于多表的联合查询之后,依照特定顺序更新各关联表数据。
网上的帖子,内容基本上都是OleDbDataAdapter和OleDbCommandBuilder的联合使用,这样使用当然很方便,但这样使用一个致命问题就是:仅能对单个表进行操作。一旦涉及到多个表,便会出现如下的问题:

一般,遇到这种情况,很多帖子就会介绍使用OleDbCommand,即直接使用SQL命令更新数据库。
但是这样做太硬核,它要满足3个苛刻条件:
1、手动开启、关闭数据库连接,
2、直接编写sql命令,
3、保证在命令执行期间一直与数据库保持连接,不能中断。
整个操作风险很高,且灵活性很差。
因此才会寻求如何单独使用OleDbDataAdapter更新Access数据库记录。
二、解决该问题前的知识储备
在了解解决方法之前,我认为先要搞懂几个问题:
2.1 OleDBDataAdapter如何处理数据?
OleDBDataAdapter可以填充DataSet,也可直接填充DataTable。
即:DataAdapter填充→DataSet→DataTable→DataRow 或者 DataAdapter填充→DataTable→DataRow
但不论那种方式,最终都要在DataRow处理数据。只有在DataRow中变更了的数据,才会在DataAdapter的Update方法执行后更新到数据库。在这里,记录的“修改”、“添加”、“删除”都认为是“变更”。
2.2 关于Table表名称
SqlDataAdapter类与OleDbDataAdapter类的Fill方法填充DataSet中表名称,默认是Table, Table1, Table2……,不是数据库中的表名。
想DataSet中表名称设置成跟数据库中表名称一样的,可以通过OleDbDataAdapter类TableMappings属性的Add方法实现。
TableMappings.Add方法,第一个参数是Fill DataSet时默认的表名,第二参数是指定的表名,可以根据需要自定义。
2.3 DataTable主键设定
在定义DataRow前,需要为DataTable指定主键,Access数据库中的主键定义传不过来。
2.4 控制更新顺序及选择特定类型DataRow
可以使用DataTable的Select方法来返回仅引用具有特定 DataRow 的 RowState 数组,然后可以将返回的 DataRow 数组传递给 Update 的 DataAdapter 方法来处理已修改的行。 通过指定要更新的行的子集,可以控制处理插入、更新和删除的顺序。
例如:处理表中“已删除”的行
adapter.Update(table.Select(null, null, DataViewRowState.Deleted));
知道上面这些,就可以着手来解决问题了。
三、完整的解决代码
3.1 定义连接字符串和sql查询语句
1 readonly OleDbConnection connection = new OleDbConnection()
2 {
3 ConnectionString = Properties.Settings.Default.myConnString
4 //myConnString内存放着Access数据库连接字符串
5 };
6 string selectCommandText = "SELECT T0101_公司情况表.公司名称_PK, T0101_公司情况表.统一社会信用代码, T0101_公司情况表.公司地址, T0301_fdc证件信息表.证件编号 FROM T0101_公司情况表 INNER JOIN T0301_fdc证件信息表 ON T0101_公司情况表.公司名称_PK = T0301_fdc证件信息表.公司名称_FK;";//基于多表的联合查询,有这种查询的时候OleDbCommandBuilder就无法使用。
3.2 单独使用OleDbDataAdapter更新Access数据库的完整代码
private void Btn_UpdateByAdapter_Click(object sender, RoutedEventArgs e)
{
/****方法2:使用OleDBDataAdapter更新数据库****/
OleDbCommand selectCommand = new OleDbCommand(selectCommandText, connection);
OleDbDataAdapter dataAdapter = new OleDbDataAdapter(selectCommand);
dataAdapter.TableMappings.Add("Table", "T0101_公司情况表");
//DataAdapter类的Fill方法填充DataSet中表名称,默认是Table, Table1, Table2……Add方法第一个参数是Fill DataSet时默认的表名,第二参数是指定的表名,可以根据需要自定义(一般定义为数据库中的表名)。
DataSet dataSet = new DataSet();
dataAdapter.Fill(dataSet);
DataTable table = dataSet.Tables["T0101_公司情况表"];
table.PrimaryKey = new DataColumn[] { table.Columns["公司名称_PK"] };//必须要指定主键
DataRow dataRow = table.Rows.Find("房县大唐盛世置业有限公司");
Random random = new Random();
dataRow["统一社会信用代码"] = "信用代码" + random.Next(1, 99);
OleDbCommand updateCommand = new OleDbCommand()
{
CommandText = "UPDATE T0101_公司情况表 SET 统一社会信用代码 = ? WHERE 公司名称_PK = ?",
Connection = connection
};
OleDbParameter parameter1 = new OleDbParameter()
{
ParameterName = "@统一社会信用代码",//虽然在Access中用途不大,但还是建议应按照查询参数的要求写。
OleDbType = OleDbType.Char,
Size = 18,
SourceColumn = "统一社会信用代码"
};
updateCommand.Parameters.Add(parameter1);
updateCommand.Parameters.Add("公司名称_PK", OleDbType.VarChar, 200, "公司名称_PK"); dataAdapter.UpdateCommand = updateCommand;
dataAdapter.Update(dataSet);
}
如何在不使用OleDbCommandBuilder情况下使用OleDbDataAdapter更新Access数据库记录的更多相关文章
- mysql-unsha1:在未知密码情况下,登录任意MYSQL数据库
摘要 这个POC用于在不知道明文密码的情况下对启用了密码安全认证插件(默认开启插件:mysql_native_password)的MYSQL数据库进行登录. 前提条件为: 1.为了获取到已知用户的ha ...
- AJAX 是一种在无需重新加载整个网页的情况下,能够更新部分网页的技术。
AJAX = Asynchronous JavaScript and XML(异步的 JavaScript 和 XML). AJAX 不是新的编程语言,而是一种使用现有标准的新方法. AJAX 是与服 ...
- java-int类型:int默认为0导致更新操作未赋值的情况下将值更新为0
日常开发中,做更新操作的时候的处理方法为:当这个字段有值则更新,没有值就不更新,在mybatis的xml中表现为: <!-- 修改记录,只修改只不为空的字段 --> <update ...
- 使用EF Model First创建edmx模型,数据库有数据的情况下,如何同时更新模型和数据库
使用"EF Model First",我们可以很直观地在edmx文件中创建数据模型,并根据模型生成数据库,整个过程快速而高效.可当数据库中有了一定的数据量,同时,可能需要对模型中字 ...
- 在不重装系统的情况下撤底删除oracle数据库及oralce的相关软件
先从控制面板删除oracle的相关应用及数据库, 删除系统变量 ORACLE_OEM_CLASSPATH=%JAVA_HOME%\lib\ext\access-bridge-64.jar;%JAVA_ ...
- 如何在不改SQL的情况下优化数据库
主题简介 在数据库运维中我们会遇到各种各样的问题,这些问题的根源可能很明显,也可能被某种表象掩盖而使我们认不清.所以运维面临的两大问题就是,第一我们没有看清本质,第二应用不允许修改.那么我们如何解决这 ...
- SQL Server 在缺少文件组的情况下如何还原数据库
SQL Server 在缺少文件组的情况下如何还原数据库 一.背景 我有一个A库,由于a,b两张表的数据量比较大,所以对表进行分区:在把A库迁移到一个新的集群上去,我只备份了A库的主分区过去进行还原为 ...
- 如何在删除ibdata1和ib_logfile的情况下恢复MySQL数据库
昨天,有个朋友对公司内部使用的一个MySQL实例开启binlog,但是在启动的过程中失败了(他也没提,为何会失败),在启动失败后,他删除了ibdata1和ib_logfile,后来,能正常启动了,但所 ...
- 排版紧凑情况下IOS 浏览器的文字部分选中问题
一.需求 一个每一项都是图文混排的列表页,在需要对其中的某一部分文字进行选中copy的时候,IOS个二货每次都是直接选中了整个列表项,无论怎么操作它的选框都没有办法做到部分选中. 这是我本周遇到遇到的 ...
随机推荐
- MySQL空间函数实现位置打卡
项目需求是跟用户当前位置判断是否在给定的地理位置范围内,符合位置限制才可以打卡,其中的位置范围是一个或多个不规则的多边形.如下图,判断用户是在清华还是北大. 图形获取区域坐标 因为项目前端使用微信小程 ...
- .Net MVC5(.Net Framework 4.0+)多语言解决方案
最近项目需要做多语言,原先是2种语言(中文/英文),现在又要加一种语言,成了3种.那么原来的方式肯定不适用了,只能升级解决方案. 原来的写法,使用三目表达式,按照当前全局变量的语言类型,返回不同的语言 ...
- java应用中的日志介绍
日志在应用程序中是非常非常重要的,好的日志信息能有助于我们在程序出现 BUG 时能快速进行定位,并能找出其中的原因. 但是,很多介绍 AOP 的地方都采用日志来作为介绍,实际上日志要采用切面的话是极其 ...
- [WPF][Rubyer] 写一个自己的 UI 库 (二) - Icon
前言 制作 WPF 的图标包,主要介绍从 iconfont 下载的图标包导入到 WPF 使用: 1. 添加文件 Ruyber 下添加 自定义控件(WPF) Icon.cs.类 IconType.cs. ...
- kolla build 配置
kolla-build.conf 配置文件: [DEFAULT] debug = false base = centos base_tag = 7.7.1908 base_arch = x86_64 ...
- linux下udev和mdev的使用
linux下设备文件系统有devfs.udev和mdev这三种. 一.devfs devfs是由Linux 2.4内核引入的,引入时被许多工程师给予了高度评价,它的出现使得设备驱动程序能自主地管理自己 ...
- CSS动画实例:行星和卫星
设页面中有<div class=" planet "></div>,用来绘制一个行星和卫星图形.这个图形包括三部分:行星.卫星和卫星旋转的轨道.定义. pl ...
- 操作系统-I/O(3)SPOOLing技术
为了缓和CPU的高速性与I/O设备低速性间的矛盾而引入了脱机输入.输出技术.该技术是利用专门的外围控制机,将低速I/O设备上的数据传送到高速磁盘上:或者相反. 事实上,当系统中引入了多道程序技术后,完 ...
- 在vmware上MBR方式安装archLinux
进入安装盘 设置好vmware的相关选项,进入下载好的系统中,显示如下 联网 输入以下命令,后面加&使其后台运行 dhcpcd & 然后ping一下检测是否联网 ping baidu. ...
- 从一个小需求感受Redis的独特魅力
分享一个简单的小需求应该怎么设计实现以及有关Redis的使用 Redis在实际应用中使用的非常广泛,本篇文章就从一个简单的需求说起,为你讲述一个需求是如何从头到尾开始做的,又是如何一步步完善的. 需求 ...