带有 IMEX 1 的 OleDBConnection 忽略时间值中的 AM / PM
OleDBConnection with IMEX 1 ignores AM / PM in time value
我使用以下代码读取了 xls 文件:
private static DataSet GetDataSetFromExcelFilePath(string filePath)
{
try
{
//Microsoft.Jet.OLEDB.4.0
using (OleDbConnection oleDbConnection = new
OleDbConnection(string.Format("Provider=Microsoft.Jet.OLEDB.4.0;Data Source={0}; Extended Properties=\"Excel 8.0;Persist Security Info=False;HDR=No;IMEX=1;\"", filePath)))
//OleDbConnection(string.Format("Provider=Microsoft.ACE.OLEDB.12.0;Data Source={0}; Extended Properties=\"Excel 8.0;Persist Security Info=False;HDR=No;IMEX=1\"", filePath)))
// using (OleDbConnection oleDbConnection = new
// OleDbConnection(string.Format("Provider=Microsoft.Jet.OLEDB.4.0;Data Source={0}; Extended Properties=\"Excel 8.0;Persist Security Info=False;HDR=No;IMEX=1\"", filePath)))
{
oleDbConnection.Open();
DataTable schema = oleDbConnection.GetOleDbSchemaTable(OleDbSchemaGuid.Tables, new object[] { null, null, null, "TABLE" });
string sheetName = schema.Rows[0].Field<string>("TABLE_NAME");
var adapter = new OleDbDataAdapter(string.Format("SELECT * FROM [{0}]", sheetName), oleDbConnection);
var dataSet = new DataSet();
adapter.Fill(dataSet, Path.GetFileName(filePath));
return dataSet;
}
}
catch (Exception ex)
{
return null;
}
}
请注意,我在连接字符串中设置了 IMEX 1,因此它将所有数据读取为字符串,但我在我客户的一台机器上遇到了一个奇怪的问题,AM / PM 或完整的 HH 被完全忽略了时间。下面是数据集是如何用上面的代码填充的:https://gyazo.com/f45c29c42b5d1339ae1fe159be4caf76 其中没有区分上午和下午。
下面是 excel 中的实际数据:https://gyazo.com/caf91c165eb08e3110fd2c3b7d4b8c51
请注意,无法更改 excel 格式或结构。最好在代码中进行任何可能的更改。
作为参考,如果有人想下载excel文件来测试这里是:https://www.dropbox.com/s/7824xh3ihlym9v9/test.xls?dl=0
当您将 Window 的语言设置为挪威时,问题可能会重现,如下所示:https://gyazo.com/d03b3056ed81e177076471a74058fdb7
ODBC 的问题是众所周知的错误。当你没有确切的 settings/office version/registry 等
时,很难给出正确的建议
您使用的IMEX=1
设置只设置了"mode",这是一种导入模式。然后您有一个注册表设置 ImportMixedTypes
,您可以在其中指定两个值:Text
或 Majority Type
。默认设置是 Text
,这就是您将所有内容都作为文本获取的原因。更多细节你可以阅读我的回答here.
您的时间问题比您尝试将其转换为文本更复杂。 excel 本身具有时间的内部表示,然后您将其转换为文本。这就是您缺少 PM/AM 部分的原因。最简单的出路是使用 24 小时格式,您可能知道但不想要这种格式。
有出路吗?是的。连接excel文件时需要直接指定列:
...
var timeColumn =
new OleDbDataAdapter("SELECT FORMAT([Time], 'hh:mm tt') as [Time] FROM [sheetName]", oleDbConnection);
...
要查看 FORMAT
的更多选项,请参阅 MSDN。
我使用以下代码读取了 xls 文件:
private static DataSet GetDataSetFromExcelFilePath(string filePath)
{
try
{
//Microsoft.Jet.OLEDB.4.0
using (OleDbConnection oleDbConnection = new
OleDbConnection(string.Format("Provider=Microsoft.Jet.OLEDB.4.0;Data Source={0}; Extended Properties=\"Excel 8.0;Persist Security Info=False;HDR=No;IMEX=1;\"", filePath)))
//OleDbConnection(string.Format("Provider=Microsoft.ACE.OLEDB.12.0;Data Source={0}; Extended Properties=\"Excel 8.0;Persist Security Info=False;HDR=No;IMEX=1\"", filePath)))
// using (OleDbConnection oleDbConnection = new
// OleDbConnection(string.Format("Provider=Microsoft.Jet.OLEDB.4.0;Data Source={0}; Extended Properties=\"Excel 8.0;Persist Security Info=False;HDR=No;IMEX=1\"", filePath)))
{
oleDbConnection.Open();
DataTable schema = oleDbConnection.GetOleDbSchemaTable(OleDbSchemaGuid.Tables, new object[] { null, null, null, "TABLE" });
string sheetName = schema.Rows[0].Field<string>("TABLE_NAME");
var adapter = new OleDbDataAdapter(string.Format("SELECT * FROM [{0}]", sheetName), oleDbConnection);
var dataSet = new DataSet();
adapter.Fill(dataSet, Path.GetFileName(filePath));
return dataSet;
}
}
catch (Exception ex)
{
return null;
}
}
请注意,我在连接字符串中设置了 IMEX 1,因此它将所有数据读取为字符串,但我在我客户的一台机器上遇到了一个奇怪的问题,AM / PM 或完整的 HH 被完全忽略了时间。下面是数据集是如何用上面的代码填充的:https://gyazo.com/f45c29c42b5d1339ae1fe159be4caf76 其中没有区分上午和下午。
下面是 excel 中的实际数据:https://gyazo.com/caf91c165eb08e3110fd2c3b7d4b8c51
请注意,无法更改 excel 格式或结构。最好在代码中进行任何可能的更改。
作为参考,如果有人想下载excel文件来测试这里是:https://www.dropbox.com/s/7824xh3ihlym9v9/test.xls?dl=0
当您将 Window 的语言设置为挪威时,问题可能会重现,如下所示:https://gyazo.com/d03b3056ed81e177076471a74058fdb7
ODBC 的问题是众所周知的错误。当你没有确切的 settings/office version/registry 等
时,很难给出正确的建议您使用的IMEX=1
设置只设置了"mode",这是一种导入模式。然后您有一个注册表设置 ImportMixedTypes
,您可以在其中指定两个值:Text
或 Majority Type
。默认设置是 Text
,这就是您将所有内容都作为文本获取的原因。更多细节你可以阅读我的回答here.
您的时间问题比您尝试将其转换为文本更复杂。 excel 本身具有时间的内部表示,然后您将其转换为文本。这就是您缺少 PM/AM 部分的原因。最简单的出路是使用 24 小时格式,您可能知道但不想要这种格式。
有出路吗?是的。连接excel文件时需要直接指定列:
...
var timeColumn =
new OleDbDataAdapter("SELECT FORMAT([Time], 'hh:mm tt') as [Time] FROM [sheetName]", oleDbConnection);
...
要查看 FORMAT
的更多选项,请参阅 MSDN。