1

有父用户控件,如下所示。

<%@ Control Language="C#" AutoEventWireup="true" CodeFile="TestUserControl.ascx.cs" Inherits="TestUserControl" %>
<%@ Register Src="~/UserControls/ChildUserControl.ascx" TagName="ChildUserControl" TagPrefix="FLI" %>
<div>    
    <FLI:ChildUserControl ID="child1" runat="server"/>    
</div>

子 usecontrol 具有 pulic 属性MatchDescription,该属性Page_Load在父控件中设置。我想根据MatchDescription属性缓存子控件的多个版本。

问题是,该MatchDescription属性不能设置在 中Page_Load,因为子控件的缓存副本一旦可用就会被使用。

我该如何解决这个问题?

谢谢!

4

1 回答 1

1

看起来使用GetVaryByCustomString是去这里的方式。我的概念证明包括以下内容:

  • WebUserControl.ascx:测试控件。它有一个公共财产MatchDescription
  • Global.asax:重写GetVaryByCustomString方法。
  • WebForm.aspx:承载控件的简单表单。

WebUserControl.ascx

将以下内容添加到控件上的标记中:

<%@ OutputCache Duration="120" VaryByParam="none" VaryByCustom="MatchDescription" %>

这指定缓存控件的持续时间(以秒为单位)并VaryByCustom="MatchDescription"指定我们将缓存的参数的名称。

WebUserControl.ascx.cs

public partial class WebUserControl1 : System.Web.UI.UserControl
{
    public string MatchDescription { get; set; }

    protected void Page_Load(object sender, EventArgs e)
    {
        object description = this.Context.Application["MatchDescription"];

        if (description != null)
        {
            this.MatchDescription = description.ToString();
        }
        else
        {
            this.MatchDescription = "Not set";
        }

        Label1.Text = "Match description: " + this.MatchDescription;
    }
}

这将检查该MatchDescription值是否存在。由于父页面中代码的工作方式,您永远不会看到“未设置”,尽管在您的实现中它可能很有用,以防该值未设置。

全球.asax

Global.asax文件添加到您的项目并添加以下方法:

public override string GetVaryByCustomString(HttpContext context, string custom)
{
    if (custom == "MatchDescription")
    {
        object description = context.Application["MatchDescription"];

        if (description != null)
        {
            return description.ToString();
        }
    }

    return base.GetVaryByCustomString(context, custom);
}

这是检查MatchDescription与缓存控件关联的位。如果没有找到该控件将正常创建。context.Application之所以使用,是因为我们需要一种方法来在父页面、用户控件和 global.asax 文件之间传递描述值。

WebForm.aspx.cs

public partial class WebForm : System.Web.UI.Page
{
    private static string[] _descriptions = new string[]
    {
        "Description 1",
        "Description 2",
        "Description 3",
        "Description 4"
    };

    protected override void OnPreInit(EventArgs e)
    {
        //Simulate service call.
        string matchDescription = _descriptions[new Random().Next(0, 4)];
        //Store description.
        this.Context.Application["MatchDescription"] = matchDescription;

        base.OnPreInit(e);
    }

    protected void Page_Load(object sender, EventArgs e)
    {
        var control = LoadControl("WebUserControl.ascx") as PartialCachingControl;
        this.Form.Controls.Add(control);

        //Indicate whether the control was cached.
        if (control != null)
        {
            if (control.CachedControl == null)
            {
                Label1.Text = "Control was cached";
            }
            else
            {
                Label1.Text = "Control was not cached";
            }
        }
    }
}

请注意,在此代码中,我正在方法中进行/模拟服务调用OnPreInit。这是必要的,因为它发生在GetVaryByCustomString方法之前的页面生命周期中。

请记住,如果控件已被缓存,Page_Load例如在方法中访问它,将需要以下形式的代码:

    if (control is PartialCachingControl &&
        ((PartialCachingControl)control).CachedControl =!= null)
{
    WebUserControl1 userControl = (WebUserControl1)((PartialCachingControl)control).CachedControl;
}

参考:

我的回答的灵感来自:有什么方法可以清除/刷新/删除 OutputCache?

Pre_Init在这个问题中找到了提示: Output Caching - GetVaryByCustomString based on value set in PageLoad()

这篇知识库文章讨论了为什么PartialCachingControl.CachedControl属性总是可以返回 null:http: //support.microsoft.com/kb/837000

于 2012-10-19T08:46:50.940 回答