首先说明使用的环境是:java和Sqlserver。

最近公司需要进行大数据量的导入操作。原来使用的是Apache POI,虽然可以实现功能,但是因为逻辑处理中需要进行许多校验,处理速度太慢,使用多线程之后也不尽如人意。在网上搜索之后,找到了OPENROWSET和OPENDATASOURCE,发现使用OPENROWSET,可以非常快速的把Excel导入到数据库中。之后的各种校验,我可以通过编写sql来实现。最终结果是6w条数据可以在10秒内完成。当然数据量增加之后,完成时间并不会明显增加。这需要编写的sql比较高效,是另一方面的问题了。

首先可能需要下载一个小的程序AccessDatabaseEngine_X64.exe。

之后需要开启配置

启用:

exec sp_configure 'show advanced options',1
reconfigure
exec sp_configure 'Ad Hoc Distributed Queries',1
reconfigure

关闭:

exec sp_configure 'Ad Hoc Distributed Queries',0
reconfigure
exec sp_configure 'show advanced options',0
reconfigure

之后可以通过OPENROWSET来查询Excel文件的内容。当然也可以改为SELECT INTO存到数据库中。

SELECT * FROM OPENROWSET('Microsoft.ACE.OLEDB.12.0',
'Excel 12.0;HDR=YES;Database=E:\DataBack\Copy of SD Expired Contracts.xlsx', ['Copy of SD Expired Contracts$'])

这样使用的前提是你知道需要导入的Sheet的名称。而如果不知道的话就需要使用下面的方法,查询出所有的Sheet名称,再由用户选择导入哪一个。

EXEC sp_addlinkedserver 'ExcelSource', '',
'Microsoft.ACE.OLEDB.12.0',
'E:\DataBack\Copy of SD Expired Contracts.xlsx',
NULL,
'Excel 8.0'
EXEC sp_addlinkedsrvlogin 'ExcelSource', 'false'
GO
EXECUTE SP_TABLES_EX 'ExcelSource'

下面是我的Java代码:

这个方法用户获得Sheet Name List。

	public static List<String> getSheetNameList(String filePath,BaseDao baseDao){
String excelSource = "ExcelSource_"+StringUtils.getUUIDString();
String addSourceSql = "{CALL SP_ADDLINKEDSERVER(?,'','Microsoft.ACE.OLEDB.12.0',?,NULL,'Excel 8.0')}";
SQLQuery query = baseDao.getSQLQuery(addSourceSql);
query.setParameter(0, excelSource);
query.setParameter(1, filePath);
query.executeUpdate();
String loginSourceSqql = "{CALL SP_ADDLINKEDSRVLOGIN(?,'false')}";
query = baseDao.getSQLQuery(loginSourceSqql);
query.setParameter(0, excelSource);
query.executeUpdate(); String sheetNameSql = "{CALL SP_TABLES_EX(?)}";
query = baseDao.getSQLQuery(sheetNameSql);
query.setParameter(0, excelSource);
query.setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP);
List<Map<String, Object>> list = query.list();
List<String> sheetList = new ArrayList<String>();
for(int i=0;i<list.size();i++){
String sheetName = (String) list.get(i).get("TABLE_NAME");
if(sheetName.endsWith("_xlnm#_FilterDatabase")){ }else{
sheetList.add(sheetName);
}
}
return sheetList;
}

这个方法用户创建一个临时表,存储Excel文件内容。临时表的字段名是根据Excel表头来创建的。

	public static String uploadAndCreateTable(String filePath,String sheetName,BaseDao baseDao) {
String importTableName = "tbl_zz_"+StringUtils.getUUIDString();
String uploadFileSql = "SELECT IDENTITY(int, 1, 1) as %s,t.*,CAST(NEWID() AS VARCHAR(36)) AS %s into %s FROM OPENROWSET('Microsoft.ACE.OLEDB.12.0','Excel 12.0;HDR=YES;Database=%s', [%s]) as t";
uploadFileSql = String.format(uploadFileSql,ImportVisitorUtil.importIndex,ImportVisitorUtil.importUUID,importTableName,filePath,sheetName);
SQLQuery query = baseDao.getSQLQuery(uploadFileSql);
query.executeUpdate();
replaceSpecialCharacter(baseDao, importTableName);
changeColumnCollation(baseDao, importTableName);
return importTableName;
}

代码编写过程中发现一个问题,如果Excel表头中含有":",在编写sql过程中会跟:name这种占位符冲突,我冒号替换成了空格。

	private static void replaceSpecialCharacter(BaseDao baseDao,String tableName){
String cha = ":";
String cha_ = "%:%";
String sql = "SELECT COLUMN_NAME columnName FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = ? AND COLUMN_NAME like ? ";
SQLQuery query = baseDao.getSQLQuery(sql);
query.setParameter(0, tableName);
query.setParameter(1, cha_);
query.setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP);
List<Map<String,Object>> list = query.list();
for(Map<String,Object> map : list){
String columnName = (String) map.get("columnName");
String newColumnName = columnName.replace(cha, " ");
String tableColumnName = String.format("%s.[%s]", tableName, columnName);
String changeSql = "{CALL SP_RENAME(?,?,'column')}";
query = baseDao.getSQLQuery(changeSql);
query.setParameter(0, tableColumnName);
query.setParameter(1, newColumnName);
query.executeUpdate();
}
}

另一个问题是,本地数据库安装的时候使用的排序规则与服务器不一致,导致编写的sql运行时出现错误,提前修改排序规则。  

	private static void changeColumnCollation(BaseDao baseDao,String tableName){
String defaultCollation = "SQL_Latin1_General_CP1_CI_AS";
String dataType = "nvarchar";
String sql = "SELECT COLUMN_NAME AS columnName,CHARACTER_MAXIMUM_LENGTH AS length FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = ? AND COLLATION_NAME <> ? AND DATA_TYPE = ? ";
SQLQuery query = baseDao.getSQLQuery(sql);
query.setParameter(0, tableName);
query.setParameter(1, defaultCollation);
query.setParameter(2, dataType);
query.setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP);
List<Map<String,Object>> list = query.list();
for(Map<String,Object> map : list){
String columnName = (String) map.get("columnName");
Integer length = (Integer) map.get("length");
String changeSql = String.format("ALTER TABLE [%s] ALTER COLUMN [%s] %s(%d) COLLATE %s",tableName,columnName,dataType,length,defaultCollation);
query = baseDao.getSQLQuery(changeSql);
query.executeUpdate();
}
}

可以使用下面sql查询出表中的所有列。

SELECT * FROM INFORMATION_SCHEMA.COLUMNS

前台可以让用户现在Excel中每一列对应的真实表的列。对应关系组织好之后,根据需要给临时表添加字段、修改数据等等,最后使用INSERT SELECT插入数据或修改数据。

编写sql的时候可能因为列名的不规范,导致sql语法错误。这时候需要在列名或表名前后添加中括号“[]”。Jaya使用String.format();比较方便。

使用OPENROWSET、Microsoft.ACE.OLEDB实现大数据量的高效导入的更多相关文章

  1. 大数据量.csv文件导入SQLServer数据库

    前几天拿到了一个400多M的.csv文件,在电脑上打开要好长时间,打开后里面的数据都是乱码.因此,做了一个先转码再导入数据库的程序.100多万条的数据转码+导入在本地电脑上花了4分钟,感觉效率还可以. ...

  2. .NET读取Excel数据,提示错误:未在本地计算机上注册“Microsoft.ACE.OLEDB.12.0”提供程序

    解决.NET读取Excel数据时,提示错误:未在本地计算机上注册“Microsoft.ACE.OLEDB.12.0”提供程序的操作: 1. 检查本机是否安装Office Access,如果未安装去去h ...

  3. 从Excel中导入数据时,提示“未在本地计算机上注册“Microsoft.ACE.OLEDB.12.0”提供程序”的解决办法

    注意,64位系统,用64位的补丁文件; https://www.cnblogs.com/A2008A/articles/2438962.html 操作系统:使用的是64位的Windows Server ...

  4. 在使用Access连接后获取数据--出现此类问题如何解决---未在本地计算机上注册“microsoft.ACE.oledb.12.0”提供程序解决办法

    转载:https://blog.csdn.net/yyzzhc999/article/details/79367114 using System; using System.Collections.G ...

  5. C#通过OLEDB导出大数据到Excel

    C#导出数据到Excel,基本有两种方法,一种方法是通过Microsoft.Office.Interop.Excel.Application,一行一列的写入Excel中:另一种方法是通过OLEDB,利 ...

  6. Microsoft.ACE.OLEDB.12.0 及其在 MSSQL中的使用

    1.Microsoft.ACE.OLEDB.12.0 简介 就是一个数据访问接口,用于在office文件和非office应用程序间传输数据.例如 Microsoft Office Access 201 ...

  7. Microsoft ACE OLEDB 12.0 数据库连接字符串

    Excel 97-2003 Provider=Microsoft.ACE.OLEDB.12.0;Data Source=c:\myFolder\myOldExcelFile.xls;Extended ...

  8. 未在本地计算机上注册“microsoft.ACE.oledb.12.0”提供程序

    这种错误的可能性有几种,比如: 1.没有安装数据访问组件,需要安装相应版本的数据访问组件: 2.没有安装相应版本的Office客户端,需要安装相应版本的Office客户端: 3.Microsoft.J ...

  9. 未在本地计算机上注册“microsoft.ACE.oledb.12.0”提供程序 解决方法

    最近在搞asp.net.今天在做数据库操作的时候,老发生错误,还以为是自己代码有问题,检查了好久都发现错误. 错误提示: 未在本地计算机上注册“microsoft.ACE.oledb.12.0”提供程 ...

随机推荐

  1. VMware/Microsoft官网查询参加的培训及认证信息

    如果你参加了VMWare的培训,会要求你拿一个已经注册的邮箱加上一个密码在VMware的系统里面登记,这样你就能在VMWARE官网查到注册,并据此你才能申请VMWare的考试认证. 例如下图,路径为 ...

  2. CSS3中-webkit-overflow-scrolling: touch 的使用方法详解

    -webkit-overflow-scrolling 属性控制元素在移动设备上是否使用滚动回弹效果. auto 使用普通滚动, 当手指从触摸屏上移开,滚动会立即停止. touch 使用具有回弹效果的滚 ...

  3. centos7使用传统网卡名

    http://serverfault.com/questions/692897/centos-7-disable-predictable-network-interface-names-during- ...

  4. js面向对象的实现(example 二)

    //这个方法和上篇文章(js面向对象的实现(example 一))中的方法类似,但是更为简洁 //通过函数赋值的方式来构造对象 //同样通过闭包的方式来封装对象及内部变量 (function () { ...

  5. python 序列化 json pickle

    python的pickle模块实现了基本的数据序列和反序列化.通过pickle模块的序列化操作我们能够将程序中运行的对象信息保存到文件中去,永久存储:通过pickle模块的反序列化操作,我们能够从文件 ...

  6. Hive的安装

    Hive的安装   第一步:解压并安装:第二步:配置 1)root用户下,解压后,改名为hive,并将hive文件夹赋给hadoop用户 tar -zxvf hive-0.9.0.tar.gz -C ...

  7. 第六百零六天 how can I 坚持(应该是六百零六天吧)

    找了个考研的借口,也是挺逗的,终于结束了,而且考的很渣. 最近发生了很多事,很快就要离开泛华了,放弃安逸,开始改变吧,其实感觉自己内心挺怂的,很怕改变,哎,这不像是有梦想,能成事的人应该有的. 还是想 ...

  8. 查找jsp页面报错技巧

    在报错跳转页面打印错误信息<div>系统执行发生错误,信息描述如下:</div> <div>错误状态代码是:${pageContext.errorData.stat ...

  9. Java7的垃圾回收

    HotSpot JVM一共有4个垃圾回收器:Serial(串行).Parallel / Throughput(并行).CMS(并发).and the new kid on the block G1(G ...

  10. mat工具MemoryAnalyzer进行分析java内存溢出hprof文件

    java服务端程序报错后会生成hprof文件,我们可以通过mat工具MemoryAnalyzer进行分析 下载地址: http://www.eclipse.org/mat/downloads.php ...