The DataSource controls described in Chapter 3, which covers the new and enhanced data access, offer the capability for integrated caching (except for the SiteMapDataSource control). The approach of these controls corresponds in principle to the ones you already know from versions 1.0 and 1.1. The data is held in memory either absolutely or sliding, depending on a specific time value. This is a possible approach, but it doesn't correspond with the practical requirements.
The new SQL Server Cache Invalidation feature will close the gap. Now you can have the caching depend on changes in an entire database table or even a single line. The latter approach will work only in combination with Yukon notifications (Yukon is the code name for the successor of SQL Server 2000, currently in Beta as well). Table-based invalidation works as well with SQL Server 2000 and even 7.0. Here, the table is polled internally with a configurable time span (standard is 60000 and minimum is 500 milliseconds) and is tested for changes.
To use SQL Server Cache Invalidation you must configure the monitored database first. This is done by the program aspnet_regsqlcache, which you'll find in the following directory:
<windir>\Microsoft.NET\Framework\<version>\
The command-line program knows various parameters. To configure the Northwind database on the local SQL server, for example, you can use the following command:
aspnet_regsqlcache -d Northwind -ed -E
In a second call of the program, you must activate the desired table:
To receive a listing of the currently activated tables, you can use the following command:
aspnet_regsqlcache -d Northwind -lt -E
Caution |
Please note that all parameters are currently case sensitive. If you have any difficulties activating the invalidation, please double-check your input. |
The program will create a new table with the (endless) name AspNet_SqlCacheTablesForChangeNotification within the specified database. For each table to be monitored, the table contains an entry with name of the table, a timestamp of the creation, and a version number. This number is incremented with each change through a trigger within the database to be monitored. By polling and comparing, the version changes can be identified and the cache gets invalidated.
CREATE TRIGGER dbo.[Customers_AspNet_SqlCacheNotification_Trigger] ON [Customers] FOR INSERT, UPDATE, DELETE AS BEGIN IF NOT EXISTS (SELECT tableName FROM dbo.AspNet_SqlCacheTablesForChangeNotification WHERE tableName = 'Customers') RETURN ELSE BEGIN UPDATE dbo.AspNet_SqlCacheTablesForChangeNotification SET changeId = changeId + 1 WHERE tableName = 'Customers' END END
Once the database, including the table, is configured, you must add both to the web.config configuration file. To allow you to do this, the new section <sqlCacheDependency> works together with the also new section <connectionStrings>. If you want to get monitoring on the Northwind database on the local SQL Server for example, your web.config file looks like this:
<configuration> <connectionStrings> <add name="Northwind" connectionString="server=.; database=Northwind;Integrated Security=SSPI;" /> </connectionStrings> <system.web> <cache> <sqlCacheDependency enabled="true" pollTime="5000"> <databases> <add name="Northwind" connectionStringName="Northwind" /> </databases> </sqlCacheDependency> </cache> </system.web> </configuration>
Although it's quite time consuming to configure the new caching possibility, it's much easier to use afterward, as the example in the following listing clearly shows. In principle, you must only activate the caching of the used SqlDataSource control and assign the desired dependency to the SqlCacheDepedency property. The dependency consists of the database name configured in the web.config file as well as the table name. Both values are separated by colon—in this case, Northwind:Customers.
<%@ page language="C#" %> <html> <head runat="server"> <title>Untitled Page</title> </head> <body> <form runat="server"> <asp:gridview id="GridView1" runat="server" allowpaging="True" datasourceid="SqlDataSource1" allowsorting="True" autogeneratecolumns="False" selectedindex="0" datakeynames="CustomerID"> </asp:gridview> <asp:sqldatasource id="SqlDataSource1" runat="server" selectcommand="SELECT CustomerID, CompanyName, ContactName FROM Customers " providername="System.Data.OleDb" connectionstring="..." sqlcachedependency="Northwind:Customers" enablecaching="True"> </asp:sqldatasource> </form> </body> </html>
You can test this example very easily by setting the polling value in the configurations file to 5 or more seconds, and afterward you can change any value in the table. If you refresh the page in the browser, the change will only become effective after a few seconds. A look at the table AspNet_SqlCacheTablesForChangeNotification shows that the version has been incremented.
You can use the new SqlCacheDependency class directly in the source code to place any elements in the cache. This is particularly useful for data that should be globally available.
Listing 11-19 shows the query and the caching of a DataSet within the cache object. An instance of the new SqlCacheDependency class is passed as a parameter when you add the object to the cache. Similar to the SqlDataSource control, the class was prepared for the Northwind database. In case of changes, the object will be invalidated in the cache, which results in a new creation of the DataSet through CacheItemRemovedCallback.
<%@ page language="C#" %> <%@ import namespace="System.Data" %> <%@ import namespace="System.Data.SqlClient" %> <script runat="server"> void Page_Load(object sender, System.EventArgs e) { if (this.Cache["ds"] == null) { this.CreateDataSet(); } if (this.IsPostBack == false) { this.LB_Customers.DataSource = this.Cache["ds"]; this.LB_Customers.DataBind(); this.LT_LastUpdate.Text = ((DateTime)this.Cache["LastUpdate"]).ToString(); } } void CreateDataSet() { string connString = ConfigurationSettings.ConnectionStrings["Yukon"]; string cmdText = "Select CompanyName FROM Customers"; SqlConnection connect = new SqlConnection(connString); SqlDataAdapter adapter = new SqlDataAdapter(cmdText, connect); DataSet ds = new DataSet(); adapter.Fill(ds); SqlCacheDependency dependency = new SqlCacheDependency("Yukon", "Customers"); this.Cache.Add("ds", ds, dependency, DateTime.MaxValue, TimeSpan.Zero, CacheItemPriority.Default, new CacheItemRemovedCallback( this.DataSetRemoveFromCache)); this.Cache.Insert("LastUpdate", DateTime.Now); } void DataSetRemoveFromCache(string key, object value, CacheItemRemovedReason reason) { this.CreateDataSet(); } </script> <html> <head runat="server"> <title>Untitled Page</title> </head> <body> <form runat="server"> <p> <asp:listbox id="LB_Customers" runat="server" height="200px" width="300px" datatextfield="CompanyName"> </asp:listbox> </p> <p> Last update: <asp:literal id="LT_LastUpdate" runat="server"> </asp:literal> </p> </form> </body> </html>
If you're using Yukon, you can pass a SqlCommand (instead of the specified name of the database and the index) to the constructor of the SqlCacheDependency class, in a way similar to this example. This way, you can use single-line-oriented SQL Cache Invalidation.
Tip |
If you want to use the SQL Server Invalidation without caching, you can use the SqlDependency class from the System.Data.SqlClient namespace. |
In previous version of ASP.NET, you already had the option to cache depending on a time span, a file, or a different cache entry using the CacheDependency class. Beyond this, the caching system couldn't be enhanced by developers, however, because the class was marked as sealed.
In the new version of the Framework, the class isn't sealed anymore and even offers some overridable methods. By inheriting from the class, you can now implement totally individual cache dependencies and integrate them seamlessly in the caching framework.
Two classes deriving from CacheDependency are already shipped with ASP.NET. You have already learned about the SqlCacheDependency class. The second class is called AggregateCacheDependency. This class allows the aggregation of several other dependencies. This way, you can, for example, make a cached object depend on two different tables. You aren't limited to a specific type; you can even mix the existing ones with custom dependencies.
Do you remember the @OutputCache directive? Of course you do! This directive allows you to specify a SQL Server dependency for complete pages as well as for controls. You can place the desired dependency through the new sqldependency attribute in the familiar form database:table. Listing 11-20 shows you how it works, and Figure 11-17 shows you how it looks.
<%@ page language="C#" %> <%@ outputcache duration="600" varybyparam="none" sqldependency="Northwind:Customers" %> <script runat="server"> void Page_Load(object sender, System.EventArgs e) { this.LT_LastUpdate.Text = DateTime.Now.ToString(); } </script> <html> <head runat="server"> <title>Untitled Page</title> </head> <body> <form runat="server"> <asp:gridview id="GridView1" runat="server" allowpaging="True" datasourceid="SqlDataSource1" allowsorting="True" autogeneratecolumns="False" selectedindex="0" datakeynames="CustomerID"> </asp:gridview> <asp:sqldatasource id="SqlDataSource1" runat="server" selectcommand="SELECT CustomerID, CompanyName, ContactName FROM Customers" providername="System.Data.OleDb" connectionstring="..."> </asp:sqldatasource> <p> Last update: <asp:literal id="LT_LastUpdate" runat="server"/> </p> </form> </body> </html>