>백엔드 개발 >C#.Net 튜토리얼 >ADO.NET은 EXCEL 구현 코드를 읽습니다((c#)).

ADO.NET은 EXCEL 구현 코드를 읽습니다((c#)).

高洛峰
高洛峰원래의
2017-01-13 16:57:161527검색

// 연결 문자열

// 连接字符串            
         string xlsPath = Server.MapPath("~/app_data/somefile.xls"); // 绝对物理路径
         string connStr = "Provider=Microsoft.Jet.OLEDB.4.0;" +
                         "Extended Properties=Excel 8.0;" + 
                         "data source=" + xlsPath;
         // 查询语句
         string sql = "SELECT * FROM [Sheet1$]";
         DataSet ds = new DataSet();
         OleDbDataAdapter da = new OleDbDataAdapter(sql, connStr);
         da.Fill(ds);    // 填充DataSet        

         // 在这里对DataSet中的数据进行操作       

         // 输出,绑定数据
         GridView1.DataSource = ds.Tables[0]; 
         GridView1.DataBind();

간단하죠? ! 다음 사항에 주의해야 한다는 점을 제외하면 모든 것이 데이터베이스 운영과 같습니다.
1. 데이터 공급자는 Jet를 사용하며 Excel 관련 속성을 설정하려면 확장 속성 키워드를 지정해야 합니다. Excel의 서로 다른 버전은 서로 다른 속성 값(확장 속성 값에 대한 유효한 Excel 버전)에 해당합니다.
Microsoft Excel 8.0(97), 9.0(2000) 및 10.0(2002) 통합 문서의 경우 Excel 8.0을 사용하세요.


Microsoft Excel 5.0 및 7.0(95) 통합 문서의 경우 Excel 5.0을 사용하세요.

Microsoft Excel 4.0 통합 문서의 경우 Excel 4.0을 사용하세요.

Microsoft Excel 3.0 통합 문서의 경우 Excel 3.0을 사용하세요.

참조:
http://msdn.microsoft.com/library/chs/default.asp?url=/library/CHS/dv_vbcode/html/vbtskcodeexamplereadingexceldataintodataset.asp

2 . 데이터 소스 경로는 물리적인 절대 경로를 사용합니다(Access와 동일)

3. 테이블 이름을 참조하는 방법은 무엇입니까?
Excel 통합 문서의 테이블(또는 범위)에 대한 유효한 참조입니다.
완전히 사용된 다양한 워크시트를 참조하려면 워크시트 이름 뒤에 달러 기호를 지정하세요. 예:

select * from [Sheet1$]
워크시트의 특정 주소 범위를 참조하려면 워크시트 이름 뒤에 달러 기호와 범위를 지정합니다. 예:

select * from [Sheet1$A1:B10]
지정된 범위를 참조하려면 범위 이름을 사용하세요. 예:

[MyNamedRange]에서 * 선택
참조:
http://msdn.microsoft.com/library/chs/default.asp?url=/library/CHS/dv_vbcode/ html /vbtskcodeexamplereadingexceldataintodataset.asp
참고:
는 Excel 통합 문서의 세 개체를 참조할 수 있습니다.
• 전체 워크시트: [Sheet1$], Sheet1은 워크시트의 이름입니다.
• 워크시트에서 Named 셀 범위: [MyNamedRange] (이름이 지정된 범위는 전체 xls에서만 고유할 수 있으므로 워크시트를 지정할 필요가 없습니다)
XLS 명명 방법: 셀 범위 선택》삽입》이름》정의
• On 워크시트 이름이 지정되지 않은 셀 범위: [Sheet1$A1:B10]
(관계형 데이터베이스에서 제공하는 다양한 개체(테이블, 뷰, 저장 프로시저 등) 중에서 Excel 데이터 소스는 테이블과 동등한 개체만 제공합니다. 워크시트와 지정된 통합 문서에 정의된 명명된 범위로 구성됩니다. 명명된 범위는 "테이블"로 간주되고 워크시트는 "시스템 테이블"로 간주됩니다.

참고:
•그렇지 않은 경우 [](대괄호)를 사용해야 합니다. 다음을 보고합니다:
FROM 절 구문 오류
• $(달러 기호)가 와야 합니다. 그렇지 않으면 다음을 보고합니다.
Microsoft Jet 데이터베이스 엔진이 'Sheet2' 개체를 찾을 수 없습니다. 개체가 존재하는지 확인하고 이름과 경로를 올바르게 작성하십시오.
•워크시트 이름이 올바르지 않거나 존재하지 않는 경우 다음과 같이 보고됩니다.
'Sheet2$'은(는) 유효한 이름이 아닙니다. 잘못된 문자나 구두점이 포함되어 있지 않은지, 이름이 너무 길지 않은지 확인하세요.
•Visual Basic 또는 VBA에서 ADO를 사용하여 Excel 데이터를 처리하는 방법에서는 [] 대신
~ 및 '(물결표 및 작은따옴표)를 사용하여 ADO를 사용할 수 있다고 언급되어 있습니다. NET 테스트가 성공하지 못했습니다.
FROM 절 구문 오류
를 보고합니다. 워크시트 이름([Sheet1$])을 참조할 때 데이터 공급자는 데이터 테이블이 왼쪽 상단의 비어 있지 않은 셀에서 시작한다고 믿습니다. 지정된 워크시트 표가 시작됩니다. 예를 들어 워크시트가 3행 C열부터 시작하고 3행 C열 이전에 1행과 2행이 모두 비어 있는 경우 3행 C열부터 시작하는 데이터만 최종 범위까지 표시됩니다. 테이블은 비어 있지 않은 단위로 끝납니다.
• 따라서 범위를 정확하게 읽어야 하는 경우 명명된 범위 [NamedRange]를 사용하거나 주소를 [Sheet1$A1:C10]으로 지정해야 합니다. ]

4. 열 이름을 인용하는 방법은 무엇입니까?
•기본 연결 문자열에 따라 데이터 공급자는 유효한 영역의 첫 번째 행을 열 이름으로 사용합니다. 이 행의 셀이 비어 있으면 F1과 F2로 표시됩니다. 여기서 서수는 다음과 같습니다. 그 뒤에 셀 이름이 옵니다. 위치는 1부터 시작하여 일관됩니다.
• 첫 번째 행을 열 이름 대신 데이터로 표시하려면 다음을 지정할 수 있습니다. 확장 속성 속성에서; 연결 문자열입니다.
기본값은 HDR=NO입니다. 형식은 다음과 같습니다. :

string connStr = "Provider=Microsoft.Jet.OLEDB.4.0;" +
"확장 속성= "Excel 8.0;HDR=NO";" +
"data source=" + xlsPath;
참고: Excel 8.0;HDR=NO에서는 큰따옴표를 사용해야 합니다(여기서 백슬래시는 C#에서 이스케이프입니다)

참조:
ms-help://MS. VSCC.v80/MS.MSDN.v80/MS.VisualStudio.v80.chs/WD_ADONET/html/745c5f95-2f02의 "Excel에 연결" 섹션 -4674-b378-6d51a7ec2490.htm
(참고: 내 MSDN에서는 이 예에서 두 개의 큰따옴표를 잘못 사용하여 테스트가 실패했습니다. 원본 텍스트는 다음과 같습니다.

확장 속성에 필요한 따옴표도 큰따옴표(
)와 함께 추가해야 합니다.

이 경우 모든 열 이름은 F로 시작하고 그 뒤에 F1, F2, F3부터 시작하는 인덱스가 옵니다. . . . . . .

5. 유효한 셀 데이터가 표시되지 않는 이유는 무엇입니까?
이런 상황이 발생할 수 있는 이유는 기본 연결에서 데이터 공급자가 이전 셀을 기반으로 후속 셀의 데이터 유형을 추론하기 때문입니다.
확장 속성에서 IMEX=1

"IMEX=1;"을 지정하여 "혼합된" 데이터 열을 항상 텍스트로 읽도록 드라이버에 알릴 수 있습니다
ref: 4와 동일

PS:在baidu这个问题的时候,有网友说,将每个单元都加上引号,这固然是格方案,但是工作量何其大啊,又不零活,庆幸自己找到”治本药方“

more ref:
如何在 Visual Basic 或 VBA 中使用 ADO 来处理 Excel 数据
http://dotnet.aspx.cc/ShowDetail.aspx?id=C673E2CD-3F1E-4919-8CE0-D69B894A0599


用程序经常需要与Excel进行数据交互,以上阐述了基于ADO.NET
读取Excel的基本方法与技巧。现在要介绍是如何动态的读取Excel数据,这里的动态指的是事先不知道Excel文件的是什么样的结构,或者无法预
测,比如一张.xls文件有多少张sheet,而且每张sheet的结构可能都不一样等等。
其实我们可以通过获取Excel的“架构信息”来动态
的构造查询语句。这里的“架构信息”与数据库领域的“数据库架构信息”意义相同(也称“元数据”),对于整个数据库,这些“元数据”通常包括数据库或可通
过数据库中的数据源、表和视图得到的目录以及所存在的约束等;而对于数据库中的表,架构信息包括主键、列和自动编号字段等。
在上文中提到

在关系数据库提供的各种对象中(表、视图、存储过程等),Excel 数据源仅提供相当于表的对象,它由指定工作簿中的工作表和定义的命名区域组成。命名区域被视为“表”,而工作表被视为“系统表”)

 

这里我们将Excel也当作一个“数据库”来对待,然后利用OleDbConnection.GetOleDbSchemaTable 方法
要获取所需的架构信息,该方法获取的架构信息与ANSI SQl-92是兼容的:


意:对于那些不熟悉 OLE DB 架构行集的人而言,它们基本上是由 ANSI SQL-92
定义的数据库构造的标准化架构。每个架构行集具有为指定构造提供定义元数据的一组列(称作 .NET
文档中的“限制列”)。这样,如果请求架构信息(例如,列的架构信息或排序规则的架构信息),则您会明确知道可以得到哪种类型的数据。如果希望了解更多信
息,请访问 Appendix B:Schema Rowsets。
ref:http://www.microsoft.com/china/msdn/library/office/office/odatanet2.mspx?mfr=true

以下是读取Excel文件内“表”定义元数据,并显示出来的的程序片断:

// 读取Excel数据,填充DataSet
         // 连接字符串            
         string xlsPath = Server.MapPath("~/app_data/somefile.xls");
         string connStr = "Provider=Microsoft.Jet.OLEDB.4.0;" +

 "Extended Properties=\"Excel 8.0;HDR=No;IMEX=1\";" + // 指定扩展属性为
 Microsoft Excel 8.0 (97) 9.0 (2000) 10.0 (2002),并且第一行作为数据返回,且以文本方式读取
                         "data source=" + xlsPath;
         string sql_F = "SELECT * FROM [{0}]";

         OleDbConnection conn = null;
         OleDbDataAdapter da = null;
         DataTable tblSchema = null;
         IList<string> tblNames = null;

         // 初始化连接,并打开
         conn = new OleDbConnection(connStr);
         conn.Open();

         // 获取数据源的表定义元数据                        
         //tblSchema = conn.GetSchema("Tables");
         tblSchema = conn.GetOleDbSchemaTable(OleDbSchemaGuid.Tables, new object[] { null, null, null, "TABLE" });

         GridView1.DataSource = tblSchema;
         GridView1.DataBind();

         // 关闭连接
         conn.Close();
 GetOleDbSchemaTable 方法的详细说明可以参考:
 http://msdn2.microsoft.com/zh-CN/library/system.data.oledb.oledbconnection.getoledbschematable.aspx

 接着是一段利用“架构信息”动态读取Excel内部定义的表单或者命名区域的程序片断:

         // 读取Excel数据,填充DataSet
         // 连接字符串            
         string xlsPath = Server.MapPath("~/app_data/somefile.xls");
         string connStr = "Provider=Microsoft.Jet.OLEDB.4.0;" +

 "Extended Properties=\"Excel 8.0;HDR=No;IMEX=1\";" + // 指定扩展属性为
 Microsoft Excel 8.0 (97) 9.0 (2000) 10.0 (2002),并且第一行作为数据返回,且以文本方式读取
                         "data source=" + xlsPath;
         string sql_F = "SELECT * FROM [{0}]";

         OleDbConnection conn = null;
         OleDbDataAdapter da = null;
         DataTable tblSchema = null;
         IList<string> tblNames = null;

         // 初始化连接,并打开
         conn = new OleDbConnection(connStr);
         conn.Open();

         // 获取数据源的表定义元数据                        
         //tblSchema = conn.GetSchema("Tables");
         tblSchema = conn.GetOleDbSchemaTable(OleDbSchemaGuid.Tables, new object[] { null, null, null, "TABLE" });

         //GridView1.DataSource = tblSchema;
         //GridView1.DataBind();

         // 关闭连接
         //conn.Close();

         tblNames = new List<string>();
         foreach (DataRow row in tblSchema.Rows) {
             tblNames.Add((string)row["TABLE_NAME"]); // 读取表名
         }

         // 初始化适配器
         da = new OleDbDataAdapter();
         // 准备数据,导入DataSet
         DataSet ds = new DataSet();

         foreach (string tblName in tblNames) {
             da.SelectCommand = new OleDbCommand(String.Format(sql_F, tblName), conn);
             try {
                 da.Fill(ds, tblName);
             }
             catch {
                 // 关闭连接
                 if (conn.State == ConnectionState.Open) {
                     conn.Close();
                 }
                 throw;
             }
         }

         // 关闭连接
         if (conn.State == ConnectionState.Open) {
             conn.Close();
         }

         // 对导入DataSet的每张sheet进行处理        
         // 这里仅做显示
         GridView1.DataSource = ds.Tables[0];
         GridView1.DataBind();

         GridView2.DataSource = ds.Tables[1];
         GridView2.DataBind();

这里我们就不需要对SELEC 语句进行“硬编码”,可以根据需要动态的构造FROM 字句的“表名”。


不仅可以,获取表明,还可以获取每张表内的字段名、字段类型等信息:

tblSchema = conn.GetOleDbSchemaTable(OleDbSchemaGuid.Columns, new object[] { null, null, null, null });

在ADO.nET
1.x 时候只有OleDb提供了GetOleDbSchemaTable
方法,而SqlClient或者OrcaleClient没有对应的方法,因为对应数据库已经提供了类似功能的存储过程或者系统表供应用程序访问,比如对
于Sql Server:

SELECT *
FROM Northwind.INFORMATION_SCHEMA.COLUMNS
WHERE TABLE_NAME = N'Customers'

而在ADO.NET 2.0中每个xxxConnenction都实现了基类System.Data.Common.DbConnection的 GetSchemal 方法
来获取数据源的架构信息。
http://msdn2.microsoft.com/zh-cn/library/system.data.common.dbconnection.getschema.aspx

//---------------------------------------------------------------------------
//IMEX 正確寫法
string connExcel = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + filePath +";Extended Properties='Excel 8.0;IMEX=1;'"; //HDR=NO;IMEX=1

using System;
 using System.Data;
 using System.Configuration;
 using System.Web;
 using System.Web.Security;
 using System.Web.UI;
 using System.Web.UI.WebControls;
 using System.Web.UI.WebControls.WebParts;
 using System.Web.UI.HtmlControls;

 using System.Data.SqlClient;
 using System.Data.OleDb;

 public partial class _Default : System.Web.UI.Page 
 {
     protected void Page_Load(object sender, EventArgs e)
     {

     }
     protected void Button1_Click(object sender, EventArgs e)
     {

         string strconn = @"Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + @"D:/last.xls" + ";Extended Properties=Excel 8.0;"; //HDR=no";//IMEX=1
         OleDbConnection conn = new OleDbConnection(strconn);
         DataSet myset = new DataSet();
         try
         {
             conn.Open();

 
             string mysql = "select * from [Sheet1$] ";//where chs <> &#39;&#39;

             OleDbDataAdapter aper = new OleDbDataAdapter(mysql, conn);

             myset.Tables.Clear();
             aper.Fill(myset, "book");

             conn.Close();

             GridView1.DataSource = myset.Tables["book"];
             GridView1.DataBind();
         }
         catch (Exception ex)
         {
             conn.Close();
             this.lb_msg.Text = ex.Message;
             return;
             // return ex.Message;
         }
     }
 }


更多ADO.NET 读取EXCEL的实现代码((c#))相关文章请关注PHP中文网!


성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.