使用 OpenRowSet 和 OpenDataSource 访问 Excel 97-2007
接口类型
有两种接口可供选择:Microsoft.Jet.OLEDB.4.0(以下简称 Jet 引擎)和Microsoft.ACE.OLEDB.12.0(以下简称 ACE 引擎)。
Jet 引擎大家都很熟悉,可以访问 Office 97-2003,但不能访问 Office 2007。
ACE 引擎是随 Office 2007 一起发布的数据库连接组件,既可以访问 Office 2007,也可以访问 Office 97-2003。
另外:Microsoft.ACE.OLEDB.12.0 可以访问正在打开的 Excel 文件,而 Microsoft.Jet.OLEDB.4.0 是不可以的。
Microsoft.ACE.OLEDB.12.0 安装文件:
1) 查看是否已經安裝此接口
進入SSMS後,在對象資源管理器中,位次展開“服務器對象”--- “鏈接服務器”---“訪問接口”,如果在訪問接口下面找到了Microsoft.ACE.OLEDB.12.0(如下圖),則說明已經安裝此接口。也有可能版本號不是12.0,此版本是對應Office 2010的。一般情況下,如果安裝了Office,此接口會跟著一起被安裝。
2) 安裝接口
如果沒能找到此接口,說明系統沒有安裝Office或者所安裝版本(32bit或64bit)與SQL Server版本不一致。可以有如下解決方案:
a) 若Office與MSSQL位數不一致,重新安裝與SQL Server位數相同的Office,同為32bit或64bit;
b) 若不準備安裝Office,可以單獨安裝Microsoft.ACE.OLEDB.12.0:32bit下載 或64bit下載,这个是office 2010版本的。
若安裝時出現點擊後閃退,可以先用WINRAR將exe安裝包解壓,再運行裏面的setup
-----------------------------
因为我们是Win7 64位系统,而且项目用到的dll都是64位的,所以要装Microsoft Access Engine-x64,但是安装的时候会检测到你机器上安装的是32位的Office,要求你把Office升级到64位,难道真的要大费周章的卸载并重装Office吗?解决方案还是有的。
- 使用"/passive"命令来安装,例如"C:\directory path\AccessDatabaseEngine_x64.exe" /passive
- 安装完成后,查看注册表HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Office\14.0\Common\FilesPaths,删除mso.dll
问题就会解决,就可以使用C# ACE来访问Access数据库。如果启动Office就会重新安装Office注意第二步。
语法一览
使用 Jet 引擎或 ACE 引擎访问,在语法上没有什么的区别。
Excel 2007 工作簿文件的扩展名是:xlsx
HDR=Yes/No
可选参数,指定 Excel 表的第一行是否列名,缺省为 Yes,可以在注册表中修改缺省的行为。
IMEX=1
可选参数,将 Excel 表中混合 Intermixed 数据类型的列强制解析为文本。
注册表设置
Microsoft.Jet.OLEDB.4.0
HKEY_LOCAL_MACHINE/SOFTWARE/Microsoft/Jet/4.0/Engines/Excel
Microsoft.ACE.OLEDB.12.0
HKEY_LOCAL_MACHINE/SOFTWARE/Microsoft/Office/12.0/Access Connectivity Engine/Engines/Excel
FirstRowHasNames
设置 HDR 参数的缺省行为,默认为 Yes。
ImportMixedTypes
设置混合列的强制解析类型,默认为文本 Text。
TypeGuessRows
设置用于解析数据类型的取样行数,默认取样前 8 行。如果设置为 0,将分析所有数据行,但不建议这样做,会影响引擎的性能。
注意:Excel 表数据列是单一列数据类型还是混合列数据类型列,是由取样行决定,而不是整列数据决定。
单一数据类型列的类型解析
Sheet1 的内容如下图所示,涵盖了大部分 Excel 的数据类型,其中 longtext 分别有 256 个 A 和 B。

对于单一数据类型列的类型解析,ACE 引擎和 Jet 引擎是一样的,下面测试 Jet 引擎的数据解析:
数据类型解析总结
- 文本:长度 <= 255,解析为 nvarchar(255),长度 > 255,解析为 ntext。
- 数值:货币解析为 money,其它均解析为 float。
- 时间:datetime。
混合数据类型列的自然解析
相对于使用 IMEX=1 的强制解析,不使用 IMEX=1,称为自然解析。下图是 Sheet2 的内容:

对于混合数据类型列的自然解析,ACE 引擎和 Jet 有细节上的区别,先看测试:
相同地方
- 取样行里数值型多于文本型,解析为 float 数值。
- 取样行里数值型少于文本型,解析为 nvarchar/ntext 文本。
- 当解析为 float 数值时,文本类型显示为 NULL,这点毫无疑问。
相异地方
- 取样行里数值型等于文本型,Jet 引擎解析为 float 数值,数值优先,ACE 引擎解析为 nvarchar/ntext 文本,文本优先。
- 当解析为 nvarchar/ntext 文本时,Jet 引擎将非文本数据显示为 NULL,ACE 引擎正确显示。
混合数据类型列的强制解析——IMEX=1
使用 IMEX=1 选参之后,只要取样数据里是混合数据类型的列,一律强制解析为 nvarchar/ntext 文本。当然,IMEX=1 对单一数据类型列的解析是不影响的。
最后一列(num<str),Jet 引擎的自然解析和强制解析都解析为 nvarchar(255),但是自然解析将数值显示为 NULL,强制解析却能正确显示,这是不一致的地方。
在数据解析的细节方面,ACE 引擎的表现优于 Jet 引擎。在前面提到的文本优先问题、非文本数据的 NULL 值问题,ACE 引擎的解析更合理。
如何解决 NULL 值问题
前 8 行(取样行)是混合数据类型的列,使用 IMEX=1 选参解决。
前 8 行是文本,8 行之外有非文本的数据,使用 ACE 引擎解决。
前 8 行是数值,8 行之外又非数值的数据:
- 将前 8 行其中一行的单元格式数字设置为文本(如果还不行,可能要手工重写该单元格,以应用文本格式,不记得是 Office 97 还是 2000 存在这个问题了);
- 修改注册表中的 TypeGuessRows(注册表设置),增加取样行数,或设置为 0 全部解析。
目的只有一个,让取样行变成混合数据类型的列,然后使用 IMEX=1 选参解决。
SQL Server 2000 中的列顺序问题
这是 SQL Server 2000 行集函数 OpenRowSet 和 OpenDataSource 本身的问题,与访问接口引擎无关,也与 Excel 版本无关。SQL Server 2005 的 OpenRowSet 和 OpenDataSource 不存在这个问题。

上图是 Sheet3 的内容,连接到 SQL Server 2000 测试看看是什么问题:
返回结果集的列顺序,是按照列名排序,并不是 Excel 表的列顺序。HDR=No 貌似正确,但仔细一看,仍然是按列名排序的。
OpenRowSet(query)
OpenRowSet(query) 可以解决这个列顺序的问题,包括后面的访问隐藏的 Sheet 或非常规命名的 Sheet,都可以用 OpenRowSet(query) 解决。
如何访问隐藏的 Sheet
隐藏 Sheet 的访问情况比较复杂,就不写测试过程了,归纳一下:
- 使用 OpenRowSet(query) 肯定可以访问。
- Excel 2007 任何写法都可以访问(Jet 引擎不能访问 Excel 2007)。
- 打开的 Excel 文件任何写法都可以访问(Jet 引擎不能访问打开的 Excel 文件)。
如何访问非常规命名的 Sheet
新建一个空白的 Sheet,重命名为 4 保存关闭:
- 使用 OpenRowSet(query) 可以正常访问。
- 其它写法,用单引号限定名称 ['4$'] 可以访问。
- OpenRowSet(query) 也可以使用单引号限定访问:'select * from [''4$'']'
引出最后一个问题,访问 Excel 97-2003 空白的 Sheet,会返回一行 NULL 值,访问 Excel 2007 空白的 Sheet,返回空结果集,数据类型均解析为 nvarchar(255)。
浙公网安备 33010602011771号