I need help.
I have a large table that looks like this.
(ID INT NOT NULL IDENTITY(1,1),PK INT , pocket VARCHAR(10))
1, 1, p1
2, 1, p2
3, 2, p3
4, 2, p4
5, 3, p5
6, 3, p6
7, 4, p7
8, 5, p1
9, 5, p2
10,5, p83
i would like to loop through the table and concatenate the pocket filed for all the records that has the same pk. and insert the pk and the concatenated string into another table in a timely manner.
can anyone help?
Emad
An easy way (but slow if you have a big table) to do this is to use a cursor. Bellow is a fully functional example.
Hope it helps!
DECLARE @.MyTable TABLE (ID INT NOT NULL,PK INT , pocket VARCHAR(10))
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (1, 1, 'p1')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (2, 1, 'p2')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (3, 2, 'p3')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (4, 2, 'p4')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (5, 3, 'p5')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (6, 3, 'p6')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (7, 4, 'p7')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (8, 5, 'p1')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (9, 5, 'p2')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (10,5, 'p83')
DECLARE @.MyResult TABLE (PK Int, SumOfPockets Varchar(4000))
DECLARE @.PK Int
DECLARE @.MyString Varchar(4000)
DECLARE cMyTable CURSOR LOCAL FAST_FORWARD FOR
SELECT PK FROM @.MyTable GROUP BY PK
OPEN cMyTable
FETCH NEXT FROM cMyTable INTO @.PK
WHILE @.@.FETCH_STATUS=0
BEGIN
SET @.MyString = ''
SELECT @.MyString = @.MyString + pocket FROM @.MyTable WHERE PK = @.PK
INSERT INTO @.MyResult(PK, SumOfPockets) VALUES (@.PK, @.MyString)
FETCH NEXT FROM cMyTable INTO @.PK
END
CLOSE cMyTable
DEALLOCATE cMyTable
SELECT * FROM @.MyResult
|||Doru,
This looks like a great solution although i have a very huge table and i ran out of resources when i tried to implement your solution. I was hopping i can do it with some loops and temp tables. do you have any other ideas.
Emad
|||i CAN'T USE LOCAL VARIABLES BECAUSE THE RESULT OF THE CONCATENATION WOULD BE MORE THAN 8000 CHAR.|||Emadkb wrote: i CAN'T USE LOCAL VARIABLES BECAUSE THE RESULT OF THE CONCATENATION WOULD BE MORE THAN 8000 CHAR.
If you're running on SQL Server 2005, you can use VARCHAR(MAX) and NVARCHAR(MAX) variables in your string concatenation.
|||we are still on SQL 2000|||Hi Emad,
If Varchar(8000) is not big enough then you'll have to use Text datatype in your destination table. This will slow down the whole thing even more :-(.
A solution that should work, but which is very slow, is:
DECLARE @.MyTable TABLE (ID INT NOT NULL,PK INT , pocket VARCHAR(10))
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (1, 1, 'p1')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (2, 1, 'p2')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (3, 2, 'p3')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (4, 2, 'p4')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (5, 3, 'p5')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (6, 3, 'p6')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (7, 4, 'p7')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (8, 5, 'p1')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (9, 5, 'p2')
INSERT INTO @.MyTable(ID, PK, pocket) VALUES (10,5, 'p83')
CREATE TABLE #MyResult (PK Int, SumOfPockets Text)
DECLARE @.PK Int, @.pocket Varchar(10), @.prev_PK Int
DECLARE @.ptrColText Varbinary(16)
DECLARE cMyTable CURSOR LOCAL FAST_FORWARD FOR
SELECT PK, pocket FROM @.MyTable
OPEN cMyTable
FETCH NEXT FROM cMyTable INTO @.PK, @.pocket
WHILE @.@.FETCH_STATUS=0
BEGIN
IF @.PK = @.prev_PK
UPDATETEXT #MyResult.SumOfPockets @.ptrColText NULL 0 @.pocket
ELSE
BEGIN
INSERT INTO #MyResult(PK, SumOfPockets)
VALUES (@.PK, @.pocket)
SELECT @.ptrColText = TEXTPTR(SumOfPockets)
FROM #MyResult WHERE PK = @.PK
SET @.prev_PK = @.PK
END
FETCH NEXT FROM cMyTable INTO @.PK, @.pocket
END
CLOSE cMyTable
DEALLOCATE cMyTable
SELECT * FROM #MyResult
|||
This looks exactly like i want. Excellent work. I will try it and let you know how it goes.
Emad
No comments:
Post a Comment