我试图自动化将CSV文件批量导入sql-server的过程,到目前为止我有这个:
----Allow for SQL to use cmd shell
--EXEC sp_configure 'show advanced options', 1 -- To allow advanced options to be changed.
--RECONFIGURE -- To update the currently configured value for advanced options.
--EXEC sp_configure 'xp_cmdshell', 1 -- To enable the feature.
--RECONFIGURE -- To update the currently configured value for this feature.
SET NOCOUNT ON
--Loop through all of the files
CREATE TABLE #tmp(excelFileName VARCHAR(100));
CREATE TABLE #tmpCol(col NVARCHAR(MAX));
INSERT INTO #tmp
EXEC xp_cmdshell 'dir /B "C:\Users\owain.esau\OneDrive\Work\Companies\IronSude\backupFiles\"';
DECLARE @fileName varchar(100)
DECLARE @tableName varchar(MAX)
DECLARE @sql varchar(MAX)
While (Select Count(*) From #tmp where excelFileName is not null) > 0
Begin
Select Top 1 @fileName = excelFileName From #tmp
SET @sql = 'BULK INSERT #tmpCol FROM "C:\Users\owain.esau\OneDrive\Work\Companies\IronSide\backupFiles\' + @filename + '" WITH (ROWTERMINATOR = '','', LASTROW = 1)'
EXEC (@sql)
UPDATE #tmpCol SET col = REPLACE(col, '"', '')
UPDATE #tmpCol SET col = REPLACE(col, ' ', '')
SET @tableName = LEFT(@filename, LEN(@filename) -4)
CREATE TABLE @tableNAme ( )
Delete from #tmp Where excelFileName = @FileName
End
DROP TABLE #tmp
DROP TABLE #tmpCol
@sql
批量插入根据需要获取列标题,但我无法弄清楚如何将其实际传递到CREATE TABLE
语句中。
非常感谢任何帮助!
-----编辑----------------------------------------- ----------
将CREATE TABLE更改为使用动态SQL后,我最终得到以下内容(我也把它放入循环中):
----Allow for SQL to use cmd shell
--EXEC sp_configure 'show advanced options', 1 -- To allow advanced options to be changed.
--RECONFIGURE -- To update the currently configured value for advanced options.
--EXEC sp_configure 'xp_cmdshell', 1 -- To enable the feature.
--RECONFIGURE -- To update the currently configured value for this feature.
SET NOCOUNT ON
--Loop through all of the files
CREATE TABLE #tmp(excelFileName VARCHAR(100));
CREATE TABLE #tmpCol(col NVARCHAR(MAX));
INSERT INTO #tmp
EXEC xp_cmdshell 'dir /B "C:\Users\owain.esau\OneDrive\Work\Companies\IronSude\backupFiles\"';
DECLARE @fileName varchar(100)
DECLARE @tableName varchar(MAX)
DECLARE @colName varchar(MAX)
DECLARE @sql varchar(MAX)
DECLARE @sql2 varchar(max)
While (Select Count(*) From #tmp where excelFileName is not null) > 0
Begin
Select Top 1 @fileName = excelFileName From #tmp
SET @sql = 'BULK INSERT #tmpCol FROM "C:\Users\owain.esau\OneDrive\Work\Companies\IronSude\backupFiles\' + @filename + '" WITH (ROWTERMINATOR = '','', LASTROW = 1)'
EXEC (@sql)
UPDATE #tmpCol SET col = REPLACE(col, '"', '')
UPDATE #tmpCol SET col = REPLACE(col, ' ', '')
WHILE (SELECT COUNT(*) FROM #tmpCol WHERE col IS NOT NULL) > 0
BEGIN
Select Top 1 @colName = col From #tmpCol
SET @tableName = LEFT(@filename, LEN(@filename) - 5)
SET @sql2 = 'CREATE TABLE [' + @tableName + ']('
SELECT @sql2 = @sql2 + '[' + col + '],' FROM #tmpCol
SET @sql2 = SUBSTRING(@sql2,1,LEN(@sql2) -1) + 'NVARCHAR(MAX)) '
EXEC(@sql2)
DELETE FROM #tmpCol WHERE col = @colName
END
Delete from #tmp Where excelFileName = @FileName
End
DROP TABLE #tmp
DROP TABLE #tmpCol
问题是,是第一个动态SQL查询:
SET @sql = 'BULK INSERT #tmpCol FROM "C:\Users\owain.esau\OneDrive\Work\Companies\IronSude\backupFiles\' + @filename + '" WITH (ROWTERMINATOR = '','', LASTROW = 1)'
由于最后一行设置为1而rowterminator为','它只拾取一列。如果我将RowTerminator更改为'\ n'它将所有内容放在一行中,如:
"Attachment Id","Attachment Owner Id","Modified By","Created By","Created Time","Modified Time","File Name",Size,"Parent Id","Attachment Type",Documents
无论如何我可以设置LASTROW ='\ n'的第一个实例?
答案 0 :(得分:2)
您可以使用动态SQL 来实现此目的:
SET @tableName = LEFT(@filename, LEN(@filename) -4)
DECLARE @strQuery VARCHAR(MAX)
SET @strQuery = 'CREATE TABLE [' + @tableName + ']('
SELECT @strQuery = @strQuery + '[' + col + '],' FROM #tmpCol
SET @strQuery = SUBSTRING(@strQuery,1,LEN(@strQuery) -1) + ') '
--If you want to execute the create table query
EXEC(@strQuery)
Delete from #tmp Where excelFileName = @FileName
字符串拆分功能
CREATE FUNCTION [dbo].[Split]
(
@String NVARCHAR(4000),
@Delimiter NCHAR(1)
)
RETURNS TABLE
AS
RETURN
(
WITH Split(stpos,endpos)
AS(
SELECT 0 AS stpos, CHARINDEX(@Delimiter,@String) AS endpos
UNION ALL
SELECT endpos+1, CHARINDEX(@Delimiter,@String,endpos+1)
FROM Split
WHERE endpos > 0
)
SELECT
'Col' = SUBSTRING(@String,stpos,COALESCE(NULLIF(endpos,0),LEN(@String)+1)-stpos)
FROM Split
)
GO
您可以尝试使用拆分功能而不是使用rowterminator = ','
拆分列行,您需要再添加一个步骤。是传递逗号分隔的字符串(首先将它分配给变量)你提供的问题,上面的函数。
功能参考
答案 1 :(得分:2)
首先构造一个包含列和DataType的字符串,然后删除最后一个","通过使用LEFT和LEN,在动态查询字符串中使用此字符串来创建所需的动态表
Declare @Columnslist varchar(max)
SELECT @Columnslist = select COLUMN_NAME+' NVARCHAR(MAX),' from TABLE_NAME for xml path('')
SELECT @Columnslist = LEFT(@Columnslist, LEN(@Columnslist) - 1)
Declare @create_cmd varchar(max)
set @create_cmd=
'If Object_id('+char(39)+@tableName+char(39)+') is Not NULL
Drop Table '+@tableName+' Create table '+@tableName+'('+@Columnslist+')'
EXEC(@create_cmd);