在现代数据库管理系统中,存储过程是一种强大的工具,它允许开发者将复杂的业务逻辑封装在数据库服务器端执行。本文将探讨如何将数据表作为输入参数传递给存储过程,并在SQL Server2008中实现增删改查操作。
在应用程序中,经常需要对数据库中的一组记录执行添加、修改和删除操作。为了简化这个过程,可以设计一个逻辑,通过一个简单的表单来实现这些操作。此外,操作完成后,还需要展示记录的详细信息。在这个过程中,数据表将作为存储过程的输入参数。
通过这种方式,表单在页面上的每次事件都不会直接与数据库交互,从而减少了服务器往返次数。所有必要的操作,如插入新记录、更新现有记录和删除现有记录,都将在客户端执行,并在用户确认后才会持久化到数据库中。这将提高应用程序的性能。
操作完成后,数据将保存在会话(Session)中。当需要数据时,可以从会话中获取。将整个数据表从会话发送到数据库进行操作,并使用存储过程来执行插入、删除和更新操作。这样可以更方便地一次性对一组行执行所有操作。
当页面首次加载时,从数据库加载数据到数据表中。在数据表中添加一个名为“Operation”的列。每当用户在网格视图上执行操作时,都会为该记录添加一个指示位到该列(插入0,更新1,删除2)。如果用户点击特定行的删除链接,则该行的操作列更新为2。在网格视图的RowDataBound事件中,隐藏了操作位为2的记录。临时表存储在会话状态中。在执行任何操作之前,将调用会话表,并对该表执行操作,然后再次将该表存储在会话中。在执行所有必要的操作后,当用户点击保存按钮时,将调用函数fnMangeOperations,该函数将过滤所有已执行操作的行,并且保存按钮将只发送需要执行操作的详细信息(可以在附加文件中的源代码中找到函数fnMangeOperations)。
private clsEmpDetails _objEmpDetails;
private DataTable _dtEmpDetails;
private DataTable _dtEmpDetailsVals;
protected void Page_Load(object sender, EventArgs e)
{
if (!IsPostBack)
{
_objEmpDetails = new clsEmpDetails();
_dtEmpDetails = _objEmpDetails.fnGetDetails();
if (_dtEmpDetails != null)
{
// Adding a new column to the table which will store the operation details
// For new insert it will store 1
// For updating the existing record it will store 0
_dtEmpDetails.Columns.Add("Operation", typeof(string));
Session["EmpDetails"] = _dtEmpDetails;
gvEmpDetails.DataSource = _dtEmpDetails;
gvEmpDetails.DataBind();
}
}
}
protected void gvEmpDetails_RowCommand(object sender, GridViewCommandEventArgs e)
{
if (e.CommandName == "AddNew")
{
try
{
_dtEmpDetails = Session["EmpDetails"] as DataTable;
string strID = (TextBox)gvEmpDetails.FooterRow.Cells[0].FindControl("txtInsID").Text;
string strName = (TextBox)gvEmpDetails.FooterRow.Cells[1].FindControl("txtInsName").Text;
string strAddress = (TextBox)gvEmpDetails.FooterRow.Cells[1].FindControl("txtInsAddress").Text;
_dtEmpDetails.Rows.Add(strID, strName, strAddress, "0");
Session["EmpDetails"] = _dtEmpDetails;
fnBindEmpDetails();
// This function will bind the data to the gridview by fetching the data from session
}
catch (Exception ex)
{
// Handle your exception
}
}
}
public string fnStoredProc(DataTable dtDetails)
{
string strMsg = "";
try
{
fnConOpen(); // Function for opening connection
SqlCommand cmdProc = new SqlCommand("spEmpDetails", con);
cmdProc.CommandType = CommandType.StoredProcedure;
cmdProc.Parameters.AddWithValue("@Type", "InsertDetails");
cmdProc.Parameters.AddWithValue("@Details", dtDetails);
cmdProc.ExecuteNonQuery();
strMsg = "Saved successfully.";
}
catch (SqlException e)
{
strMsg = "Data not saved successfully.";
strMsg = e.Message.ToString();
}
finally
{
fnConClose(); // Function for closing connection
}
return strMsg;
}
首先创建一个与前端匹配的表类型的参数。在这种情况下:
CREATE TYPE EmpType AS TABLE (
ID INT,
Name VARCHAR(3000),
Address VARCHAR(8000),
Operation SMALLINT
)
ALTER PROCEDURE spEmpDetails
@Type VARCHAR(15),
@Details EmpType READONLY
AS
BEGIN
IF (@Type='FetchDetails')
SELECT * FROM EmployeeDetails
ELSE
BEGIN
-- For deleting the details from the table
DELETE FROM EmployeeDetails WHERE ID IN (SELECT ID FROM @Details WHERE Operation=2)
-- For updating the details in the table
UPDATE e SET e.Name=d.Name, e.Address=d.Address FROM EmployeeDetails e, @Details d WHERE d.ID=e.ID and d.Operation=1
-- For inserting the new records in the table
INSERT INTO EmployeeDetails(ID, Name, Address) SELECT ID, Name, Address FROM @Details WHERE Operation=0;
END
END
GO