ASP.NET(C#) 讀取EXCEL問題彙總

使用OLEDB可以對excel文件進行讀取,我們只要把該excel文件作爲數據源即可。

一 在D盤創建excel文件test.xls:

  

二 將工作表Sheet1的內容讀取到DataSet
  string strConn = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=D:/test.xls;"+
      "Extended Properties='Excel 8.0'";
  DataSet ds = new DataSet();
  OleDbDataAdapter oada = new OleDbDataAdapter("select * from [Sheet1$]", strConn);
  oada.Fill(ds);

 

讀取的DataSet爲:

  

從圖中可以看出excel文件中的第一行變成了DataSet中的列名,這正是系統的默認設置。

如果想把第一行也作爲數據行,那我們可以給連接字符串添加一個HDR=No屬性如:
  string strConn = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=D:/test.xls;"+
      "Extended Properties='Excel 8.0;HDR=No'";
  DataSet ds = new DataSet();
  OleDbDataAdapter oada = new OleDbDataAdapter("select * from [Sheet1$]", strConn);
  oada.Fill(ds);
  結果也許會讓你有點想不到:

  

第一行的第一列和第三列都變成空的了,這是因爲系統把第一列識別成了數字,把第三列識別成了日期,

而第一行的數據不符合格式的要求,所以就變成空的了。

四 我們還可以把所有列都做爲字符串來讀取,只要添加屬性IMEX=1即可

  string strConn = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=D:/test.xls;"+
      "Extended Properties='Excel 8.0;HDR=No;IMEX=1'";
  DataSet ds = new DataSet();
  OleDbDataAdapter oada = new OleDbDataAdapter("select * from [Sheet1$]", strConn);
  oada.Fill(ds);
  結果又會如何呢?

  

是不是再次出乎你的意料,第三行的日期怎麼變成數字了,其實excel在轉換格式的時候就自動把日期變成數字了,

那這個數字是怎麼來的呢 ? 如果你把日期改成1900年1月1日,那麼你可以看到他的轉換結果是1,以此類推,39902是哪一天就明白了吧。

這裏解決辦法:

方法一:
  public static string getDateStr(string strValue)
      {
          int i = Convert.ToInt32(strValue);
          DateTime d1 = Convert.ToDateTime("1900-1-1");
          DateTime d2 = d1.AddDays(i - 2);
          string strTemp = d2.ToString("d");

          return strTemp;
      }
  方法二:
  DateTime.FromOADate(Convert.ToInt32(strValue)).ToString("d");

五 也許你並不想讀取整個excel的內容

如果只想讀取前兩列可以用:select * from [Sheet1$A:B]

如果只想讀取A1到B2的內容,就用:select * from [Sheet1$A1:B2]

六 如果不知道工作表的名字或名字被人爲修改了該怎麼辦呢?

我們可以通過索引來獲取指定工作表的名字,以下方法可以用來獲取工作表名稱的數組:


  ArrayList al = new ArrayList();
  string strConn;
  strConn = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=D:/test.xls;"+
      "Extended Properties=Excel 8.0;";
  OleDbConnection conn = new OleDbConnection(strConn);
  conn.Open();
  DataTable sheetNames = conn.GetOleDbSchemaTable
      (OleDbSchemaGuid.Tables, new object[] { null, null, null, "TABLE" });
  conn.Close();
  foreach (DataRow dr in sheetNames.Rows)
  {
      al.Add(dr[2]);
  }
  return al;

 

IMEX=1的時候並不是全都會作爲字符串來處理,根據系統的默認設置,通常如果前8行有字符串,則該列會作爲字符串來處理,如果全都爲數字,則該列爲數字列,日期也是一樣。

如果你覺得8行不夠或者太多了,則只能修改註冊表HKEY_LOCAL_MACHINE/Software/Microsoft/Jet/4.0/Engines/Excel/TypeGue***ows,如果此值爲0,則會根據所有行來判斷使用什麼類型,通常不建議這麼做,除非你的數據量確實比較少。 

 

無法讀取EXCEL中的數據單元格。有數據,但是讀出來全是空值。

解決方法:

1.在導入數據連接字符串中,將IMEX=1加入,“Provider=Microsoft.Jet.OLEDB.4.0;Data Source="C:\Data.xls";Extended Properties="Excel 8.0;HDR=Yes;IMEX=1; ”,這樣就可以。

注:

“HDR=Yes;”指示第一行中包含列名,而不是數據;

“IMEX=1;”通知驅動程

序始終將“互混”數據列作爲文本讀取。

兩者必須一起使用。

本以爲這樣就OK了。但在實際使用過程中,這樣設置還是不行,查閱了不少資料才發現,原來還有一個註冊表裏的信息需要修改,這樣帶能讓excel不再使用前8行的內容來確定該列的類型。

註冊表修改內容如下:

在HKEY_LOCAL_MACHINE\Software\Microsoft\Jet\4.0\Engines\Excel有一個TypeGue***ows值,預設是8,表示會先讀取前8列來決定每一個欄位的型態,所以如果前8列的資料都是數字,到了第9列以後出現的文字資料都會變成null,所以如果要解決這個問題,只要把TypeGue***ows機碼值改成0,就可以解這個問題了。

 

asp.net 解決使用OLEDB導入excel數據時同時包含文本和數字的列無法正常讀取情況


1.連接語句中必須包含IMEX=1

         EXCEL2003:Provider=Microsoft.Jet.OLEDB.4.0;Data Source='文件路徑';Extended Properties=Excel 8.0;HDR=Yes;IMEX=1

         EXCEL2007:Provider=Microsoft.ACE.OLEDB.12.0;Data Source='文件路徑';Extended Properties=Excel 12.0;HDR=Yes;IMEX=1

 2.設置註冊表信息:【HKEY_LOCAL_MACHINE/Software/Microsoft/Jet/4.0/Engines/Excel/】

        ImportMixedType :Text (注:值爲Text,則該列爲文本;值爲Majority Type,則取數據多的類型)

        TypeGue***ows :0  (注:0表示要讀取所有數據再來判斷是否是混合類型)

OLEDB 連接 Excel 中的 HDR 與 IMEX 解釋

 Microsoft.Jet.OleDb 連接 Excel,就像數據庫一樣操作 Excel,以下是一個示例的連接字符串:

string connString = "Provider=Microsoft.Jet.OleDb.4.0;Data Source=" + filePath +";Extended Properties='Excel    8.0;HDR=NO;IMEX=1;'";

HDR

HDR 表示第一行是否是標題行。

  • 若爲 YES,則第一行是標題行(即列名稱),不是數據;

  • 若爲 NO,則第一行不是標題行,跟後面的行一樣,是數據。

IMEX

即 intermixed,表示混合數據類型時如何處理。Excel 不像 Access 樣,Access 每一字段(列)具有數據類型,Excel 不具有,所以 Excel 第一行第一列可以存儲字符串,第二行第一列又可以存儲數字……同樣的列,存儲不同的數據類型,這就形成了混合數據類型。

如果我們的 Excel 不存在混合數據類型,則可以省略 IMEX;如果我們的 Excel 存在混合數據類型,則需要正確指定 IMEX,否則 OLEDB 很可能錯誤地判斷數據類型,導致讀取出來的數據是空白,甚至讀取不到行等錯誤。

  • 若爲 0,則爲輸出模式,此情況下只能用作寫入 Excel;

  • 若爲 1,則爲輸入模式,此情況下只能用作讀取 Excel,並且始終將 Excel 數據作爲文本類型讀取;

  • 若爲 2,則爲連接模式,此情況下既可用作寫入、也可用作讀取。

所以若要讀取混合數據類型,應該將 IMEX 設置爲 1;若誤設置爲 0,則讀取不到任何行;若誤設置爲 2 或省略,則有些數據讀取出來是空白。

注意:輸出模式對應寫入、輸入模式對應讀取。

 Microsoft.Jet.OleDb 連接 Excel,就像數據庫一樣操作 Excel,以下是一個示例的連接字符串:

string connString = "Provider=Microsoft.Jet.OleDb.4.0;Data Source=" + filePath +";Extended Properties='Excel 8.0;HDR=NO;IMEX=1;'";

HDR

HDR 表示第一行是否是標題行。

  • 若爲 YES,則第一行是標題行(即列名稱),不是數據;

  • 若爲 NO,則第一行不是標題行,跟後面的行一樣,是數據。

IMEX

即 intermixed,表示混合數據類型時如何處理。Excel 不像 Access 樣,Access 每一字段(列)具有數據類型,Excel 不具有,所以 Excel 第一行第一列可以存儲字符串,第二行第一列又可以存儲數字……同樣的列,存儲不同的數據類型,這就形成了混合數據類型。

如果我們的 Excel 不存在混合數據類型,則可以省略 IMEX;如果我們的 Excel 存在混合數據類型,則需要正確指定 IMEX,否則 OLEDB 很可能錯誤地判斷數據類型,導致讀取出來的數據是空白,甚至讀取不到行等錯誤。

  • 若爲 0,則爲輸出模式,此情況下只能用作寫入 Excel;

  • 若爲 1,則爲輸入模式,此情況下只能用作讀取 Excel,並且始終將 Excel 數據作爲文本類型讀取;

  • 若爲 2,則爲連接模式,此情況下既可用作寫入、也可用作讀取。

所以若要讀取混合數據類型,應該將 IMEX 設置爲 1;若誤設置爲 0,則讀取不到任何行;若誤設置爲 2 或省略,則有些數據讀取出來是空白。

注意:輸出模式對應寫入、輸入模式對應讀取。

 Microsoft.Jet.OleDb 連接 Excel,就像數據庫一樣操作 Excel,以下是一個示例的連接字符串:

string connString = "Provider=Microsoft.Jet.OleDb.4.0;Data Source=" + filePath +";Extended Properties='Excel 8.0;HDR=NO;IMEX=1;'";

HDR

HDR 表示第一行是否是標題行。

  • 若爲 YES,則第一行是標題行(即列名稱),不是數據;

  • 若爲 NO,則第一行不是標題行,跟後面的行一樣,是數據。

IMEX

即 intermixed,表示混合數據類型時如何處理。Excel 不像 Access 樣,Access 每一字段(列)具有數據類型,Excel 不具有,所以 Excel 第一行第一列可以存儲字符串,第二行第一列又可以存儲數字……同樣的列,存儲不同的數據類型,這就形成了混合數據類型。

如果我們的 Excel 不存在混合數據類型,則可以省略 IMEX;如果我們的 Excel 存在混合數據類型,則需要正確指定 IMEX,否則 OLEDB 很可能錯誤地判斷數據類型,導致讀取出來的數據是空白,甚至讀取不到行等錯誤。

  • 若爲 0,則爲輸出模式,此情況下只能用作寫入 Excel;

  • 若爲 1,則爲輸入模式,此情況下只能用作讀取 Excel,並且始終將 Excel 數據作爲文本類型讀取;

  • 若爲 2,則爲連接模式,此情況下既可用作寫入、也可用作讀取。

所以若要讀取混合數據類型,應該將 IMEX 設置爲 1;若誤設置爲 0,則讀取不到任何行;若誤設置爲 2 或省略,則有些數據讀取出來是空白。

注意:輸出模式對應寫入、輸入模式對應讀取。


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章