MARS:Multiple Active Result Sets,多重作用結果集。
MARS能在同一條資料連線(SqlConnection)開啟一個以上的結果集,並讓你同時存取這些結果集。
讓我們回憶一下,我們撰寫資料庫的流程,
- 連線字串(SqlConnection)
- SQL定義(SqlCommand)
- 取回資料(DataReader, DataSet)
- 處理(...DataBind...)
- 關閉連線(Close)
不管怎麼寫,大多跳不出這幾個步驟,而如果同一個畫面,同時需要存取幾個DataReader,DataSet,同樣的東西你就要寫兩次,寫三次,這樣是非常沒有效率,這時就非常合適使用MARS技巧。也就是說,步驟1裡的SqlConnection可以同時間給多個SqlCommand來使用,或是說,你需要同時存取同一台Database Server裡的不同資料庫,那也非常合適使用這個MARS技巧。
MARS的設定也異常簡單,只需要在設定連線字串時加入「
MultipleActiveResultSets=True」來啟用MARS功能即可。
我們來看一個實例,我們要輸出一個Orders資料表,然後在Orders裡又抓取Orders Details的資料來一起輸出。
無MARS的GridView查詢
以下範例我使用一個下拉式選單,來查詢北風Customers的資料,同時顯示兩個GridView。
GridView.aspx
02 | Imports System.Data.SqlClient |
03 | Imports System.Configuration |
06 | Inherits System.Web.UI.Page |
08 | Protected Sub Page_Load( ByVal sender As Object , ByVal e As System.EventArgs) Handles Me .Load |
09 | If Not Page.IsPostBack Then |
10 | Dim ddlconn As New SqlConnection(ConfigurationManager.ConnectionStrings( "NorthwindConnectionString" ).ConnectionString) |
12 | Dim ddlcmd As New SqlCommand |
13 | ddlcmd.CommandText = "select distinct ContactTitle from customers" |
14 | ddlcmd.CommandType = CommandType.Text |
15 | ddlcmd.Connection = ddlconn |
16 | ddlcmd.Connection.Open() |
18 | Dim ddldr As SqlDataReader |
19 | ddldr = ddlcmd.ExecuteReader |
21 | ddlTitle.DataSource = ddldr |
23 | ddlTitle.DataTextField = "ContactTitle" |
25 | ddlTitle.DataValueField = "ContactTitle" |
33 | Protected Sub ddlTitle_SelectedIndexChanged( ByVal sender As Object , ByVal e As System.EventArgs) Handles ddlTitle.SelectedIndexChanged |
35 | Dim conn As New SqlConnection(ConfigurationManager.ConnectionStrings( "NorthwindConnectionString" ).ConnectionString) |
37 | Dim cmd As New SqlCommand |
38 | cmd.CommandText = "select top 5 * from customers where ContactTitle=@ContactTitle" |
39 | cmd.CommandType = CommandType.Text |
43 | Dim ContactParam As New SqlParameter |
44 | ContactParam.ParameterName = "@ContactTitle" |
46 | ContactParam.Value = ddlTitle.SelectedValue |
47 | cmd.Parameters.Add(ContactParam) |
50 | Dim dr As SqlDataReader |
52 | dr = cmd.ExecuteReader(CommandBehavior.CloseConnection) |
73 | Dim dtconn As New SqlConnection(ConfigurationManager.ConnectionStrings( "NorthwindConnectionString" ).ConnectionString) |
75 | Dim dtcmd As New SqlCommand |
76 | dtcmd.CommandText = "select top 5 * from customers where ContactTitle=@dtContactTitle order by customerid desc" |
77 | dtcmd.CommandType = CommandType.Text |
78 | dtcmd.Connection = dtconn |
81 | Dim dtContactParam As New SqlParameter |
82 | dtContactParam.ParameterName = "@dtContactTitle" |
84 | dtContactParam.Value = ddlTitle.SelectedValue |
85 | dtcmd.Parameters.Add(dtContactParam) |
87 | Dim dap As New SqlDataAdapter |
88 | Dim dt As New DataTable |
89 | dap.SelectCommand = dtcmd |
92 | gvDT.DataSource = dt.DefaultView |
你可以看到在SelectedIndexChanged事件裡,我為了要讓兩個GridView顯示不同的資料,所以我必須
開關三次SqlConnection(含下拉式控制項的那一次),但其實這是不必要的,接下來我們來看看MARS的使用。
Multiple Active Result Sets
MARS.aspx
1 | <%@ Page Language= "VB" %> |
2 | <%@ Import Namespace = "System.Data" %> |
3 | <%@ Import Namespace = "System.Data.SqlClient" %> |
4 | <%@ Import Namespace = "System.Configuration" %> |
01 | Dim conn As New SqlConnection(ConfigurationManager.ConnectionStrings( "NorthwindConnectionString" ).ConnectionString) |
03 | Protected Sub Page_Load( ByVal sender As Object , _ |
04 | ByVal e As System.EventArgs) |
06 | Dim cmd As New SqlCommand |
07 | Dim OrdersReader As SqlDataReader |
10 | " SELECT TOP 5 Customers.CompanyName, Customers.ContactName, " & _ |
11 | " Orders.OrderID, Orders.OrderDate, " & _ |
12 | " Orders.RequiredDate, Orders.ShippedDate " & _ |
13 | " FROM Orders, Customers " & _ |
14 | " WHERE Orders.CustomerID = Customers.CustomerID " & _ |
15 | " ORDER BY Customers.CompanyName, Customers.ContactName " |
17 | cmd.CommandType = CommandType.Text |
21 | OrdersReader = cmd.ExecuteReader() |
23 | gvOrders.DataSource = OrdersReader |
30 | Protected Sub gvOrders_RowDataBound( ByVal sender As Object , _ |
31 | ByVal e As System.Web.UI.WebControls.GridViewRowEventArgs) |
33 | Dim OrderRecord As IDataRecord |
34 | Dim lblOrderDetail As Label |
38 | OrderRecord = CType (e.Row.DataItem, IDataRecord) |
42 | lblOrderDetail = CType (e.Row.FindControl( "lblOrderDetail" ), Label) |
44 | If OrderRecord Is Nothing Or lblOrderDetail Is Nothing Then |
48 | Dim cmd As New SqlCommand |
49 | Dim OrderDetailReader As SqlDataReader |
51 | "SELECT Products.ProductName, [Order Details].UnitPrice, " & _ |
52 | " [Order Details].Quantity, [Order Details].Discount " & _ |
53 | " FROM [Order Details], Products " & _ |
54 | " WHERE [Order Details].ProductID = Products.ProductID " & _ |
55 | " AND [Order Details].OrderID = " + _ |
56 | Convert.ToString(OrderRecord( "OrderID" )) |
57 | cmd.CommandType = CommandType.Text |
64 | OrderDetailReader = cmd.ExecuteReader() |
66 | While OrderDetailReader.Read() |
67 | lblOrderDetail.Text += OrderDetailReader(0).ToString() + " |
79 | <form id= "form1" runat= "server" > |
81 | <asp:Label ID= "lblCounter" runat= "server" > |
02 | <asp:GridView ID= "gvOrders" runat= "server" AutoGenerateColumns= "False" OnRowDataBound= "gvOrders_RowDataBound" Width= "100%" > |
04 | <asp:BoundField HeaderText= "Company Name" DataField= "CompanyName" ></asp:BoundField> |
05 | <asp:BoundField HeaderText= "Contact Name" DataField= "ContactName" ></asp:BoundField> |
11 | <asp:Label ID= "lblOrderDetail" runat= "server" > |
14 | <asp:BoundField HeaderText= "Order Date" DataField= "orderdate" DataFormatString= "{0:d}" > |
16 | <asp:BoundField HeaderText= "Required Date" DataField= "requireddate" DataFormatString= "{0:d}" > |
18 | <asp:BoundField HeaderText= "Shipped Date" DataField= "shippeddate" DataFormatString= "{0:d}" > |
範例的重點在71裡RowDataBound事件裡使用
同一個SqlConnection物件來連接。MARS在你需要同一Web Form裡需要使用越來越多結果集時,你就可以發現MARS的效果,讓我們的程式更有彈性,更靈活。
請問一下 MultipleActiveResultSets 和 Connection Pool 是一樣的意思嗎
回覆刪除有沒有什麼差異
謝謝