Sql 取决于字符串的长度。是否要将数据从#TEMP1表插入到#TEMP2表…这就是您要执行的操作?只需更改~符号而不是
Sql 取决于字符串的长度。是否要将数据从#TEMP1表插入到#TEMP2表…这就是您要执行的操作?只需更改~符号而不是,sql,sql-server,Sql,Sql Server,取决于字符串的长度。是否要将数据从#TEMP1表插入到#TEMP2表…这就是您要执行的操作?只需更改~符号而不是
取决于字符串的长度。是否要将数据从#TEMP1表插入到#TEMP2表…这就是您要执行的操作?只需更改
~
符号而不是在我的回答中,很抱歉,对于xml和其他不需要的东西,您有交叉应用程序。我没有重写或复制粘贴您的解决方案。这是我自己写的。这只是有趣的一半。我想知道这会有多快?它需要在字符串上扫描大量次。请您发表评论,以便我可以改进我的answer@MartinBrown-将其转换为行后,每行将保存很少的数据,而不是我和我所保存的整个字符串。首先,您要进行拆分,扫描字符串一次。然后,对于拆分的每个部分,您将执行一次拆分,再次扫描字符串。然后对每个部分执行两次替换和一个ParseName,该ParseName将扫描每个部分三次。总共要扫描整个字符串五次。此外,还将创建大量中间字符串,这些字符串都需要内存分配,如果内存被碎片化,可能会很慢。虽然如果只有一个字符串,这不会是一个问题,但是如果您有几百万个字符串要处理,那么这将是一个问题。
DECLARE @str VARCHAR(max)='88.47,1,263759,10.00| 303.53,2,264051,13.00| 147.92,3,264052,6.00| 43.26,4,268394,10.00| 127.7,5,269229,4.00|'
SELECT Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)')))
FROM (SELECT Cast ('<M>' + Replace(@str, '|', '</M><M>') + '</M>' AS XML) AS Data) AS A
CROSS APPLY Data.nodes ('/M') AS Split(a)
SELECT Id,c1,c2,c3
FROM (SELECT Id=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 4), ';', '.'),
C1=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 3), ';', '.'),
c2=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 2), ';', '.'),
c3=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 1), ';', '.')
FROM (SELECT Cast ('<M>' + Replace(@str, '|', '</M><M>') + '</M>' AS XML) AS Data) AS A
CROSS APPLY Data.nodes ('/M') AS Split(a)) a
WHERE id IS NOT NULL
SELECT c1,c2,c3,c4
FROM (SELECT C1=Replace(Parsename(Replace(Replace(col, '.', ';'), ',', '.'), 4), ';', '.'),
C2=Replace(Parsename(Replace(Replace(col, '.', ';'), ',', '.'), 3), ';', '.'),
C3=Replace(Parsename(Replace(Replace(col, '.', ';'), ',', '.'), 2), ';', '.'),
C4=Replace(Parsename(Replace(Replace(col, '.', ';'), ',', '.'), 1), ';', '.')
FROM (SELECT Split.a.value('.', 'VARCHAR(100)') col
FROM (SELECT Cast ('<M>' + Replace(@str, '|', '</M><M>') + '</M>' AS XML) AS Data) AS A
CROSS APPLY Data.nodes ('/M') AS Split(a))v) a
WHERE c1 IS NOT NULL;
create table #test(string varchar(8000))
insert into #test values
('88.47,1,263759,10.00| 303.53,2,264051,13.00| 147.92,3,264052,6.00| 43.26,4,268394,10.00| 127.7,5,269229,4.00|'),
('88.47,1,263759,10.00| 303.53,2,264051,13.00| 147.92,3,264052,6.00| 43.26,4,268394,10.00| 127.7,5,269229,4.00|')
SELECT c1,c2,c3,c4
FROM (SELECT C1=Replace(Parsename(Replace(Replace(col, '.', ';'), ',', '.'), 4), ';', '.'),
C2=Replace(Parsename(Replace(Replace(col, '.', ';'), ',', '.'), 3), ';', '.'),
C3=Replace(Parsename(Replace(Replace(col, '.', ';'), ',', '.'), 2), ';', '.'),
C4=Replace(Parsename(Replace(Replace(col, '.', ';'), ',', '.'), 1), ';', '.')
FROM (SELECT Split.a.value('.', 'VARCHAR(100)') col
FROM (SELECT Cast ('<M>' + Replace(string, '|', '</M><M>') + '</M>' AS XML)
AS Data from #test) AS A
CROSS APPLY Data.nodes ('/M') AS Split(a))v) a
WHERE c1 IS NOT NULL;
SELECT REPLACE(PARSENAME(REPLACE(REPLACE(ColumnName, '.', '~'), ',', '.'), 4), '~', '.'),
REPLACE(PARSENAME(REPLACE(REPLACE(ColumnName, '.', '~'), ',', '.'), 3), '~', '.'),
REPLACE(PARSENAME(REPLACE(REPLACE(ColumnName, '.', '~'), ',', '.'), 2), '~', '.'),
REPLACE(PARSENAME(REPLACE(REPLACE(ColumnName, '.', '~'), ',', '.'), 1), '~', '.')
From #TEMP
CREATE FUNCTION dbo.parseData ( @stringToSplit VARCHAR(MAX) )
RETURNS
@return TABLE (ID int, Column1 real, Column2 int, Column3 int, Column4 real)
AS
BEGIN
DECLARE @char char;
DECLARE @len int = LEN(@stringToSplit);
DECLARE @buffer varchar(50) = '';
DECLARE @field int = 1;
DECLARE @Column1 real
DECLARE @Column2 int
DECLARE @Column3 int
DECLARE @Column4 real
DECLARE @row int = 1
DECLARE @i int = 1;
WHILE @i <= @len BEGIN
SELECT @char = SUBSTRING(@stringToSplit, @i, 1)
IF @char = ','
BEGIN
IF @field = 1
SET @Column1 = CONVERT(real, @buffer);
ELSE IF @field = 2
SET @Column2 = CONVERT(int, @buffer);
ELSE IF @field = 3
SET @Column3 = CONVERT(int, @buffer);
SET @buffer = '';
SET @field = @field + 1
END
ELSE IF @char = '|'
BEGIN
SET @Column4 = CONVERT(real, @buffer);
INSERT INTO @return (ID, Column1, Column2, Column3, Column4)
VALUES (@row, @Column1, @Column2, @Column3, @Column4);
SET @buffer = '';
SET @row = @row + 1
SET @field = 1
END
ELSE
BEGIN
SET @buffer = @buffer + @char
END
SET @i = @i + 1;
END
RETURN
END
GO
SELECT Col1 = '88.47,1,263759,10.00| 303.53,2,264051,13.00| 147.92,3,264052,6.00| 43.26,4,268394,10.00| 127.7,5,269229,4.00|'
INTO #Temp1;
INSERT INTO #Temp1
VALUES ('88.47,1,263759,10.00| 303.53,2,264051,13.00| 147.92,3,264052,6.00| 43.26,4,268394,10.00| 127.7,5,269229,4.00|')
SELECT data.*
INTO #Temp2
FROM #Temp1 CROSS APPLY parseData(#Temp1.Col1) as data
SELECT *
FROM #Temp2
DROP TABLE #Temp1
DROP TABLE #Temp2
-- First two queries to check the results are the same.
-- Note the Parsename technique returns strings rather than reals which is why
-- the last column has .00 at the end of the numbers in the Parsename tecnique.
DECLARE @str VARCHAR(max)='88.47,1,263759,10.00| 303.53,2,264051,13.00| 147.92,3,264052,6.00| 43.26,4,268394,10.00| 127.7,5,269229,4.01|'
SELECT c1,c2,c3, c4
FROM (SELECT C1=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 4), ';', '.'),
C2=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 3), ';', '.'),
C3=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 2), ';', '.'),
C4=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 1), ';', '.')
FROM (SELECT Cast ('<M>' + Replace(@str, '|', '</M><M>') + '</M>' AS XML) AS Data) AS A
CROSS APPLY Data.nodes ('/M') AS Split(a)) a
WHERE c1 IS NOT NULL;
SELECT *
FROM dbo.parseData(@str)
GO
-- Now lets time the Parsename method over 10,000 itterations
SET NOCOUNT ON;
DECLARE @str VARCHAR(max)='88.47,1,263759,10.00| 303.53,2,264051,13.00| 147.92,3,264052,6.00| 43.26,4,268394,10.00| 127.7,5,269229,4.00|'
DECLARE @i int = 0
declare @table table (c1 decimal, c2 int, c3 int, c4 decimal)
DECLARE @Start datetime = GETDATE();
while @i < 1000
begin
INSERT INTO @table
SELECT c1,c2,c3, c4
FROM (SELECT C1=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 4), ';', '.'),
C2=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 3), ';', '.'),
C3=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 2), ';', '.'),
C4=Replace(Parsename(Replace(Replace(Rtrim(Ltrim(Split.a.value('.', 'VARCHAR(100)'))), '.', ';'), ',', '.'), 1), ';', '.')
FROM (SELECT Cast ('<M>' + Replace(@str, '|', '</M><M>') + '</M>' AS XML) AS Data) AS A
CROSS APPLY Data.nodes ('/M') AS Split(a)) a
WHERE c1 IS NOT NULL;
DELETE FROM @table;
set @i = @i + 1;
end
DECLARE @End datetime = GETDATE()
PRINT CONVERT(nvarchar(50),@Start,126) + ' - ' + convert(nvarchar(50),@End,126) + ' - ' + convert(nvarchar(50), DATEDIFF(ms, @start, @end))
GO
-- Now the my technique over 10,000 itterations
SET NOCOUNT ON;
DECLARE @str VARCHAR(max)='88.47,1,263759,10.00| 303.53,2,264051,13.00| 147.92,3,264052,6.00| 43.26,4,268394,10.00| 127.7,5,269229,4.00|'
DECLARE @i int = 0
declare @table table (c1 decimal, c2 int, c3 int, c4 decimal)
DECLARE @Start datetime = GETDATE();
while @i < 1000
begin
INSERT INTO @table
SELECT *
FROM dbo.parseData(@str)
DELETE FROM @table;
set @i = @i + 1;
end
DECLARE @End datetime = GETDATE()
PRINT CONVERT(nvarchar(50),@Start,126) + ' - ' + convert(nvarchar(50),@End,126) + ' - ' + convert(nvarchar(50), DATEDIFF(ms, @start, @end))
GO