C#中SqlDataAdapter的使用小结

本文涉及的产品
RDS SQL Server Serverless,2-4RCU 50GB 3个月
推荐场景:
云数据库 RDS SQL Server,基础系列 2核4GB
简介:

SqlDataAdapter概述

SqlDataAdapter是 DataSet和 SQL Server之间的桥接器,用于检索和保存数据。SqlDataAdapter通过对数据源使用适当的Transact-SQL语句映射 Fill(它可更改DataSet中的数据以匹配数据源中的数据)和 Update(它可更改数据源中的数据以匹配 DataSet中的数据)来提供这一桥接。当SqlDataAdapter填充 DataSet时,它为返回的数据创建必需的表和列(如果这些表和列尚不存在)。

)我们可以通过以下三种方法来创建SqlDataAdapter对象:

使用方法

 

1、通过连接字符串和查询语句 

[csharp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. string strConn="uid=账号;pwd=密码;database=数据库;server=服务器";//SQL Server链接字符串  
  2. strSql="SELECT * FROM 表名";  
  3.    
  4. SqlDataAdapter da=new SqlDataAdapter(strSql,strConn);  
  5. DataSet ds=new DataSet();//创建DataSet实例  
  6. da.Fill(ds,"自定义虚拟表名");//使用DataAdapter的Fill方法(填充),调用SELECT命令  
 

这种方法有一个潜在的缺陷。假设应用程序中需要多个SqlDataAdapter对象,用这种方式来创建的话,会导致创建每个SqlDataAdapter时,都同时创建一个新的SqlConnection对象,方法二可以解决这个问题

 

2、通过查询语句和SqlConnection对象来创建

[csharp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. string strConn="uid=账号;pwd=密码;database=数据库;server=服务器";//SQL Server链接字符串  
  2.  SqlConnection conn=new SqlConnection(strConn);  
  3.    
  4. string strSql="SELECT * FROM 表名";  
  5.  SqlDataAdapter da = new SqlDataAdapter(strSql, conn);  
  6. DataSet ds=new DataSet();//创建DataSet实例  
  7. da.Fill(ds,"自定义虚拟表名");//使用DataAdapter的Fill方法(填充),调用SELECT命令  

 

3、通过SqlCommand对象来创建

 

[csharp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. string strConn="uid=账号;pwd=密码;database=数据库;server=服务器";//SQL Server链接字符串    
  2. SqlConnection connSql=new SqlConnection (strConn); //Sql链接类的实例化    
  3. connSql.Open ();//打开数据库    
  4. //使用SqlDataAdapter时没有必要从Connection.open()打开,    
  5. //SqlDataAdapter会自动打开关闭它。    
  6. string strSql = "SELECT * FROM 表名"//要执行的SQL语句    
  7. SqlCommand cmd=new SqlCommand(strSql,connsql);  
  8. SqlDataAdapter da=new SqlDataAdapter(cmd); //创建DataAdapter数据适配器实例    
  9. DataSet ds=new DataSet();//创建DataSet实例    
  10. da.Fill(ds,"自定义虚拟表名");//使用DataAdapter的Fill方法(填充),调用SELECT命令    
  11. ConnSql.Close ();//关闭数据库   


SqlDataAdapter da=new SqlDataAdapter(strSQL,ConnSql); //创建DataAdapter数据适配器实例DataSet ds=new DataSet();//创建DataSet实例da.Fill(ds,"自定义虚拟表名");//使用DataAdapter的Fill方法(填充),调用SELECT命令ConnSql.Close ();//关闭数据库

注意

 

如果只需要执行SQL语句或SP,就没必要用到DataAdapter ,直接用SqlCommandExecute系列方法就可以了。sqlDataadapter的作用是实现DatasetDB之间的桥梁:比如将对DataSet的修改更新到数据库

SqlDataAdapterUpdateCommand的执行机制是:当调用SqlDataAdapter.Update()时,检查DataSet中的所有行,然后对每一个修改过的Row执行SqlDataAdapter.UpdateCommand ,也就是说如果未修改DataSet中的数据,SqlDataAdapter.UpdateCommand不会执行。

使用要点

1、SqlDataAdapter内部通过SqlDataReader获取数据,而默认情况下SqlDataReader不能获知其查询语句对应的数据库表名,

所以下面的代码:

[csharp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. string strConn = "uid=账号;pwd=密码;database=数据库;server=服务器";//SQL Server链接字符串    
  2. strSql="SELECT * FROM 表名";    
  3.    
  4. SqlDataAdapter da = new SqlDataAdapter(strSql,strConn);  
  5. DataSet ds = new DataSet();  
  6. da.Fill(ds);  

 

会在DataSet中创建一个新的DataTable,这个新的DataTable会拥有名为CustomerIDCompanyName 列,但是DataTable对象的名称是Table,而不是我们希望的Customers

 

这个问题,可以通过添加TableMapping来解决: 

[csharp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. string strConn="uid=账号;pwd=密码;database=数据库;server=服务器";//SQL Server链接字符串    
  2. strSql="SELECT * FROM 表名";    
  3.    
  4. SqlDataAdapter da=new SqlDataAdapter(strSQL,strConn);  
  5. da.TableMappings.Add("Table""Customers");   // 设置对象名称  
  6. DataSet ds=new DataSet();  
  7. da.Fill(ds);  
其实最简洁的方法是通过使用 Fill 方法的重载,通过指定 DataTable ,像这样:

SqlDataAdapter.Fill(DataSet,"MyTableName");

这样就可以不必使用TableMappings集合。

 

2、在使用Fill方式时,可以指定DataTable,而不是DataSet

[csharp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. string strConn="uid=账号;pwd=密码;database=数据库;server=服务器";//SQL Server链接字符串    
  2. strSql="SELECT * FROM 表名";    
  3.    
  4. SqlDataAdapter da = new SqlDataAdapter(strSql, strConn);  
  5. DataTable tbl=new DataTable( );  
  6. da.Fill(tbl);  

3、注意打开和关闭连接的处理

 

在调用SqlCommand对象执行sql命令之前,需要保证与该对象关联的SqlConnection对象时打开的,否则SqlCommand的方法执行时将引发一个异常,但是我们在上面的代码中看到,SqlDataAdapter没有这样的要求。

 

如果调用SqlDataAdapterFill方法,并且其SelectCommand属性的SqlConnection是关闭状态,则SqlDataAdapter会自动打开它,然后提交查询,获取结果,最后关闭连接。如果在调用Fill方法前,SqlConnection是打开的,则查询执行完毕后,SqlConnection还将是打开的,也就是说SqlDataAdapter会保证SqlConnection的状态恢复到原来的情形。

 

这有时会导致性能问题,需要注意,例如下面的代码:

[csharp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1.    
  2. string strConn="uid=账号;pwd=密码;database=数据库;server=服务器";//SQL Server链接字符串    
  3. SqlConnection conn=new SqlConnection(strConn);  
  4. SqlDataAdapter daCustomers,daOrders;  
  5.    
  6. strSql="SELECT * FROM Customers";  
  7.  daCustomers = new SqlDataAdapter(strSql, conn);  
  8.    
  9. strSql="SELECT * FROM Orders";  
  10.  daOrders=new SqlDataAdapter(strSql, conn);  
  11.    
  12. DataSet ds=new DataSet();  
  13.    
  14. daCustomers.Fill(ds,"Customers");  
  15.  daOrders.Fill(ds,"Orders");  


 

以上代码会导致连接被打开和关闭两次,在调用Fill方法时各一次。为了避免打开和关闭SqlConnection对象,在调用SqlDataAdapter对象的Fill方法之前,我们可以先打开SqlConnection对象,如果希望之后关闭连接,我们可以再调用Close方法,就像这样:

 

[csharp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. cn.Open();  
  2.    
  3. daCustomers.Fill(ds,"Customers");  
  4.  daOrders.Fill(ds,"Orders");  
  5.    
  6. cn.Close();  


 

4、多次调用Fill方法需要注意数据重复和有效更新数据的问题

 

[csharp] view plain copy print ? 在CODE上查看代码片 派生到我的代码片
  1. string strConn="uid=账号;pwd=密码;database=数据库;server=服务器";//SQL Server链接字符串    
  2.  strSql="SELECT * FROM Customers";  
  3.    
  4. SqlDataAdapter da=new SqlDataAdapter(strSql, strConn);  
  5.  DataSet ds=new DataSet();  
  6.  da.Fill(ds,"Customers");  
  7.    
  8. //…….  
  9.    
  10. da.Fill(ds,"Customers");  


 

我们分析上面的代码,通过两次调用Fill方法,SqlDataAdapter执行两次查询,并两次将查询结果保存到DataSet中,第一次调用在DataSet中创建了一个名为Customers的新表。第二次调用Fill方法将查询的结果追加到DataSet中的同一个表中,因此,每个客户的信息将在DataSet中出现两次!当然,如果数据库管理员对Customers表定义了主键,则SqlDataAdapter在天成DataTable时,会判断重复行,并自动丢弃掉旧的值。

 

考虑一下,假定一个特定客户在第一次调用Fill方法时,存储于数据库中,那么SqlDataAdapter会将其添加到新建的DataTable中。如果后来这个客户被删除了,那么第二次调用Fill方法时,SqlDataAdapter将不会在查询结果中找到该客户信息,但是它也不会将客户信息从DataSet中删除。这就导致了数据更新的问题。

 

所以推荐的做法是,在调用Fill方法前,先删除本地DataSet中缓存的数据!


转载:http://blog.csdn.net/gatieme/article/details/20695853

相关实践学习
使用SQL语句管理索引
本次实验主要介绍如何在RDS-SQLServer数据库中,使用SQL语句管理索引。
SQL Server on Linux入门教程
SQL Server数据库一直只提供Windows下的版本。2016年微软宣布推出可运行在Linux系统下的SQL Server数据库,该版本目前还是早期预览版本。本课程主要介绍SQLServer On Linux的基本知识。 相关的阿里云产品:云数据库RDS SQL Server版 RDS SQL Server不仅拥有高可用架构和任意时间点的数据恢复功能,强力支撑各种企业应用,同时也包含了微软的License费用,减少额外支出。 了解产品详情: https://www.aliyun.com/product/rds/sqlserver
目录
相关文章
|
C# 索引 Windows
Winform控件优化之TabControl控件的使用和常用功能
TabControl是一个分页切换(tab)控件,不同的页框内可以呈现不同的内容,将主要介绍调整tab的左右侧显示、设置多行tab、禁用或删除tabpage、隐藏TabControl头部的选项卡等
5297 0
Winform控件优化之TabControl控件的使用和常用功能
|
C# 前端开发
WPF - 图形设计器(Diagram Designer)
原文:WPF - 图形设计器(Diagram Designer)   OpenExpressApp计划中包括建模工具,计划是采用MetaEdit+模型来作为元模型,使用codeproject的《WPF Diagram Designer》一系列文章来做为设计器实现参考,本篇介绍一下codeprojcet的这四个文章,推荐给对图形设计器感兴趣的人去看看,通过WPF的模板功能和其他功能可以很方便的设计出图形编辑器。
3598 0
|
JavaScript 关系型数据库 MySQL
盘点6个.NetCore+Vue前后端分离的开源项目
盘点6个.NetCore+Vue前后端分离的开源项目
1512 0
|
C# C++
创建目标类型对象在C#7.3中不可用,请使用9.0或更高的语言版本
创建目标类型对象在C#7.3中不可用,请使用9.0或更高的语言版本
1802 0
创建目标类型对象在C#7.3中不可用,请使用9.0或更高的语言版本
如何将DataGridView中的滚动条显示出来
1、选项设置   DataGridView有一个ScrollBars,可以设置水平滚动条,垂直滚动条,或者两种包含。   2、实际运行效果
5335 0
|
8月前
|
存储 JSON 前端开发
利用json-server快速在本地搭建一个JSON服务
利用json-server快速在本地搭建一个JSON服务
296 0
|
前端开发 JavaScript Java
Layui实现动态选项卡(超详细)
Layui实现动态选项卡(超详细)
225 0
|
8月前
|
SQL Java 数据库
慢SQL的治理经验
在当今的数字化时代,数据库已经成为企业不可或缺的核心组件。然而,随着数据量的不断增加和查询的复杂性提高,慢查询成为了数据库性能的瓶颈之一。慢SQL不仅会影响系统的响应速度,还可能导致数据丢失或损坏,给企业带来巨大的损失。因此,慢SQL的治理成为了数据库管理的重要任务之一。本文将分享一些慢SQL的治理经验,包括如何识别、分析和优化慢查询。通过了解慢查询的原因和解决方法,我们可以提高数据库的性能和稳定性,为企业的业务发展提供更好的支持。
313 2
|
8月前
|
前端开发 JavaScript C#
10款值得推荐的Blazor UI组件库
10款值得推荐的Blazor UI组件库
362 0
|
SQL 缓存 关系型数据库
MySQL explain 中的 rows 究竟是如何计算的?
今天同事在处理系统慢SQL时遇到几个疑惑的问题,简单描述如下~
MySQL explain 中的 rows 究竟是如何计算的?