Passing an array or DataTable into a stored procedure
Passing an array or DataTable into a stored procedure
Introduction
Have you ever wanted to pass a DataTable, or array, containing tens or tens of thousands (or even millions) of records, into a SQL Server2000 stored procedure in just one database call? Ever wanted to pass a list of IDs of records to select or delete into a SP? Then read on.
The trick is to convert lists of data into byte arrays, which are then passed as Image type parameters into the stored procedure, then in the SP the image is transformed into a table variable containing the original data. The following SQL server function is one of a number of functions I have written to easily transform an Image parameter into a variable table. This one creates a table of varchars.
Collapse | Copy Code
CREATE FUNCTION dbo.GetTableVarchar(@Data image)
RETURNS @DataTable TABLE (RowID int primary key IDENTITY ,
Value Varchar(8000))
AS
BEGIN
--First Test the data is of type Varchar.
IF(dbo.ValidateExpectedType(103, @Data)<>1) RETURN
--Loop thru the list inserting each
-- item into the variable table.
DECLARE @Ptr int, @Length int,
@VarcharLength smallint, @Value Varchar(8000)
SELECT @Length = DataLength(@Data), @Ptr = 2
WHILE(@Ptr<@Length)
BEGIN
--The first 2 bytes of each item is the length of the
--varchar, a negative number designates a null value.
SET @VarcharLength = SUBSTRING(@Data, @ptr, 2)
SET @Ptr = @Ptr + 2
IF(@VarcharLength<0)
SET @Value = NULL
ELSE
BEGIN
SET @Value = SUBSTRING(@Data, @ptr, @VarcharLength)
SET @Ptr = @Ptr + @VarcharLength
END
INSERT INTO @DataTable (Value) VALUES(@Value)
END
RETURN
END
I have also created a number of C# classes for creating byte arrays from collects of different types of data.
Calls to the database, in the following examples, use SQLHelper from the Microsoft Application Blocks, as these hide most of the dull plumbing required to call a SP. MSDN Website.
Please bear with my humble snail farm examples; I couldn t bring myself to use customer/orders examples.
We ll start off with a simple case for the first example.
Passing a list of integer IDs to select records with
Code to take the checked items in a checked list box and pass their IDs in to a stored procedure. The SP then returns a result set containing the details for each selected ID.
Collapse | Copy Code
private DataSet GetSelectedSnailDetails()
{
SQLIntListCreator snailIDs = new SQLIntListCreator();
foreach(Snail aSnail in mySelectionCheckedListBox.CheckedItems)
snailIDs.AddValue(aSnail.SnailID);
return SQLHelper.ExecuteDataset(CONN_STRING,
"GetSnailDetails", snailIDs.GetList());
}
Stored procedure to take a list of ints in the form of an image (byte array), convert it to a table variable of ints which is then joined on the SnailID column of the Snails table to return details for each selected snail.
Collapse | Copy Code
CREATE PROCEDURE GetSnailDetails
@SnailIDs image
AS
SELECT Snail.*
FROM dbo.GetTableInt(@SnailIDs) SIDs
INNER JOIN Snails ON SIDs.Value = Snails.SnailID
GO
Stored procedure to delete all the snails in the @Snails image list..
Collapse | Copy Code
CREATE PROCEDURE DeleteSnails
@SnailIDs image
AS
DELETE Snail
FROM dbo.GetTableInt(@SnailIDs) SIDs
INNER JOIN Snails ON SIDs.Value = Snails.SnailID
GO
Passing a collection of snail statuses to a SP for updating
Collapse | Copy Code
private void UpdateSnailStatuses(Snail[] snails)
{
SQLIntListCreator snailIDs = new SQLIntListCreator();
SQLVarcharListCreator statuses =
new SQLVarcharListCreator();
SQLRowversionListCreator rowVersions=
new SQLRowversionListCreator();
foreach(Snail aSnail in snails)
{
snailIDs.AddValue(aSnail.SnailID);
statuses.AddValue(aSnail.Status);
rowVersions.AddValue(aSnail.RowVersion);
}
SQLHelper.ExecuteNonquery(CONN_STRING,
"UpdateSnailStatuses", snailIDs.GetList(),
statuses.GetList(), rowVersions.GetList());
}
Stored procedure to update the statuses of snails while also checking for the correct timestamp rowversion.
Collapse | Copy Code
CREATE PROCEDURE dbo.UpdateSnailStatuses
@SnailIDs image,
@Statuses image,
@RowVersions image
AS
UPDATE Snail
SET Snail.Status = Statuses.Value
FROM Snail
INNER JOIN GetTableInt(@SnailIDs) SnailIDs
ON Snail.SnailID = SnailIDs.Value
INNER JOIN GetTableVarchar(@Statuses) Statuses
ON SnailIDs.RowID = Statuses.RowID
INNER JOIN GetTableRowversion(@RowVersions) RowVersions
ON Statuses.RowID = RowVersions.RowID
WHERE TSEQUAL(Snail.RowVersion, RowVersions.Value)
RETURN @@ERROR
Passing a DataTable to a SP for insertion into a table
Code to convert the data in a DataTable into 5 byte arrays, which are then passed to the AddSnails stored procedure to be inserted into the Snail table.
Collapse | Copy Code
private void AddSnails(DataTable snails)
{
SQLVarcharListCreator species = new SQLVarcharListCreator();
SQLDatetimeListCreator dOB = new SQLDatetimeListCreator();
SQLNvarcharListCreator location = new SQLNvarcharListCreator();
SQLBitListCreator isMale = new SQLBitListCreator();
SQLMoneyListCreator value = new SQLMoneyListCreator();
foreach(DataRow row in snails.Rows)
{
species.AddValue(row["Species"]);
dOB.AddValue(row["DOB"]);
location.AddValue(row["Location"]);
isMale.AddValue(row["IsMale"]);
value.AddValue(row["Value"]);
}
//A command is explicitly created here
//instead of using SQLHelper as we need to be
// able to set the CommandTimeout property
//big enough to insert 1 000 000 records.
using(SqlConnection cnn = new SqlConnection(CONN_STRING))
{
cnn.Open();
SqlCommand comInst = new SqlCommand("AddSnails", cnn);
comInst.CommandType = CommandType.StoredProcedure;
comInst.CommandTimeout = 10000;
comInst.Parameters.Add("@Species", species.GetListAndReset());
comInst.Parameters.Add("@DOB", dOB.GetListAndReset());
comInst.Parameters.Add("@Location", location.GetListAndReset());
comInst.Parameters.Add("@IsMale", isMale.GetListAndReset());
comInst.Parameters.Add("@Value", value.GetListAndReset());
comInst.ExecuteNonQuery();
}
}
Stored procedure to convert the images into table variables, join them together and insert the resulting data into the Snail table.
Collapse | Copy Code
ALTER PROCEDURE dbo.AddSnails
(
@Species image,
@DOB image,
@Location image,
@IsMale image,
@Value image
)
AS
INSERT INTO Snail (Species, DOB, Location, IsMale, Value)
SELECT Species.Value, DOB.Value,
Location.Value, IsMale.Value, Value.Value
FROM GetTableMoney(@value) Value
INNER JOIN GetTableBit(@isMale) IsMale ON Value.RowID = IsMale.RowID
INNER JOIN GetTableVarchar(@species) Species
ON IsMale.RowID = Species.RowID
INNER JOIN GetTableDateTime(@dOB) DOB ON Species.RowID = DOB.RowID
INNER JOIN GetTableNvarchar(@location) Location
ON DOB.RowID = Location.RowID
RETURN @@ERROR
Design view of the join across all the table variables.
Performance
I ran a performance comparison between inserting snail records using the above stored procedure and a standard 1 record insert stored procedure called many times. These are the results I got.
Records Inserted Standard one stored procedure call per record (msec) One stored procedure call for all records (msec)
1 2 10
10 22 14
100 219 60
1000 2 259 505
10 000 22 593 4 083
100 000 301 368 44 645
1 000 000 3 094 385 571 020
The comparison was done on a PC running a P4 1.8 GHz processor with 512 MB RAM. This is a very crude comparison as network speed, table structure and other things will greatly influence the speed of both methods. In this example both the SQL Server and the client where running on the same machine resulting in a very fast network. Inserts into the snail table are fast as it has only one index (clustered).
Points of interest
• Strong Type checking: Each SQLListCreator object prefixes the byte array it produces with a data type byte e.g. 107 for the SQLMoneyListCreator. The SQL function then checks this first byte and throws an error if it is of the wrong data type. As RAISERROR cannot be called from within a SQL function, the following work around is used. An attempt to convert the error message to an int is made; this results in an error being returned like this - 'Syntax error converting the varchar value'. The real error is: 'Function expected a list of Int types but was given a list of Money types.' to a column of data type int. Not the most elegant message but it points you in the right direction.
• ToString: ToString() returns a hex version of the list, this can be useful during development/debugging when you may want to paste lists into Query Analyzer. E.g.
Collapse | Copy Code
SELECT C.Value, PV.Value, EC.Value, TS.Value,
P.Value, PV.Value/TS.Value
FROM GetTableVarchar
(0x670007416C20476F7265000E47656F726765
20572E2042757368000B52616C7068204E61646572) C
INNER JOIN GetTableInt(0x65030A32590301E5C2002BFD8B)
PV ON C.RowID = PV.RowID
INNER JOIN GetTableInt(0x650000010A0000010F00000000) EC ON
PV.RowID = EC.RowID
INNER JOIN GetTableMoney
(0x6B00000117782C6150000001B0E20FACF00000001217F259D0)
TS ON EC.RowID = TS.RowID
INNER JOIN GetTableBit(0x6C9B) P ON TS.RowID = P.RowID
• Data Types: I have implemented SQL functions and C# classes for BigInt, Int, Bit, Datetime, Money, RowVersion, Uniqueidentifier, Varchar and Nvarchar. Most other data types could be implemented without too much difficulty, the exceptions are ntext, text and image. These data types can contain more that 8000 bytes and as such the SQL Substring function can not be used to strip the individual items from a list of items contained in an image.
• Nulls: Null values can be added into a list with the AddNull() method, also the AddValue(object value) overload will accept DBNull and any object that implements INullable and returns true for IsNull().
• Packet size: The database connection packet size can be set to a larger size than the default in the connection string. This may give additional performance improvements when sending large amounts of data to the database in single calls, but will degrade performance for smaller calls.
License
This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.
Bạn đang đọc truyện trên: AzTruyen.Top