有2种方式来实现缓存部分页。
片段缓存:这种情况下,你把确定要缓存的内容,包裹在一个专用的用户控件里,然后只需要对这个控件做输出缓存就行。
post-cache substitution 缓存后替换。在这种情况下,你对动态变化的内容不进行缓存,只需要用Substitution控件替换这个动态内容就行。

这两种方式,片段缓存实现起来是最简单。然而,使用哪种方式取决于你要缓存的内容的数量。如果你有个小的区域要缓存,片段缓存很合适。相反的,如果你仅仅有一个小的区域是动态内容,不需要缓存,那么post-cache substitution 缓存后替换。可能更合适。两种方式提供相同的性能。

最灵活的实现部分缓存的方法是,不用输出缓存,而是使用数据缓存在代码中程序化的处理缓存。

实现片段缓存,你只需要创建一个用户控件,设置这个用户控件的OutputCache 指令,就可以了。这样的话,整个页面不会被缓存,但是这个用户控件会被缓存。片段缓存在概念上和页面缓存一样的。

post-cache substitution以一个方法为中心,这方法已经被加入到HttpResponse类中。这个方法是WriteSubstitution().它接受一个参数。一个委托回调方法(HttpResponseSubstitutionCallback).这回掉方法返回部分页面内容。

当asp.net页面框架收到一个缓存的页面,它会自动的触发你的回掉方法来获取动态内容。然后将你的内容插入到缓存的html页面中。好处是即使你的页面还没被缓存,ASP.net仍然可以调用你的回调方法获取动态内容。本质上,这个方法就是你的方法被调用,产生动态内容,可以保证你方法总是被调用,并且他的内容不会被缓存。这个生成动态内容的方法要是static,因为asp.net需要调用改方法,即使没有实例的情况下。(很显然,让你的页面是从缓存中来,你的页面没有被创建)。这个方法的签名也很直接,接受一个HttpContext 对象来表示当前的请求,返回一个string作为新的html。

看如下的例子:

protected void Page_Load(object sender, EventArgs e)
       {
           Response.Write(DateTime.Now.ToString());

           Response.WriteSubstitution(GetDate);

           Response.WriteSubstitution((x) => "
" + DateTime.Now.ToString() + ""); //可以直接使用拉姆达表达式,简化写法。

       }

       private static string GetDate(HttpContext context)
       {
           return "
" + DateTime.Now.ToString() + "";
       }

WriteSubstitution接受的参数为HttpResponseSubstitutionCallback委托,该委托接受一个HttpContext 对象作为参数。返回string。

使用的时候,必须将方法定义为static,不然会报错。

上面的代码中,如果指定了OutputCache指令,则除了Response.WriteSubstitution方法不会被缓存外,其他方法都会被缓存。

也就是说在调试的时候,response.write会被忽略,因为从缓存发给客户端,而Response.WriteSubstitution仍然被执行。

事实上,ASP.net提供了实现缓存后替换的快捷方式,采用Substitution 控件(在控件栏里可以找到),拖拽到页面既可以。

使用时,你不能看到改控件的内容。记住,缓存后替换只允许执行静态方法。ASP.net会跳过页面生命周期,这意味着它不会创建任何控件对象,或者触发控件事件。如果你的动态内容基于值或者其他控件,你可能要使用其他技术,比如数据缓存,因为这些控件不能在回调函数中使用。