9

谁能解释为什么不能在 ASP.Net 的服务器控件声明中使用内联代码块?

下面是一个简单的例子...

....
<form id="form1" runat="server">
    <asp:Label ID="Label1" runat="server" Text="<%= SomeProperty %>"></asp:Label>
</form>
....

代码块按字面意思呈现到页面...

<span id="Label1"><%= SomeProperty %></span>

我最初的想法是它与页面生命周期中处理这些事情的顺序有关。据我了解,这些<%=...%>块相当于Response.Write(...)代码隐藏中的代码。而且由于服务器控件实际上并未按照标记中的声明进行渲染,我想在此渲染发生之前可能无法处理嵌入式代码块。

我将非常感谢任何人能更好地解释这一点。

然而,数据绑定代码块<%#...%>的行为方式显然不同,但谁能告诉我为什么可以将这些嵌入到服务器控件中......

....
<asp:Repeater id=Repeater1 runat="server">
    ....
    <ItemTemplate>
        <asp:Label ID="Label1" runat="server" Text='<%# Eval(“SomeProperty”) %>'></asp:Label>
    </ItemTemplate>
    ....
</asp:Repeater>
....

这工作正常。

4

3 回答 3

7

您对<%=...%>语法的看法基本正确。这是引擎盖下发生的事情的示例:

<script runat="server">
    public string SomeProperty { get { return "Hello World!"; } }
</script>

<form id="form1" runat="server">
    <%= SomeProperty %>
    <div>
        <asp:Label ID="Label1" runat="server" Text="<%= SomeProperty %>"></asp:Label>
    </div>
</form>

对其进行解析并创建以下 C# 代码(我已对其进行了一些简化):

private Label @__BuildControlLabel1() 
{
    Label @__ctrl = new Label();
        
    this.Label1 = @__ctrl;
    @__ctrl.ApplyStyleSheetSkin(this);
    @__ctrl.ID = "Label1";
    @__ctrl.Text = "<%= SomeProperty %>";
    return @__ctrl;
}

private void @__Renderform1(HtmlTextWriter @__w, Control parameterContainer) 
{
    @__w.Write( SomeProperty );
    @__w.Write("\r\n    <div>\r\n        ");
    parameterContainer.Controls[0].RenderControl(@__w);
    @__w.Write("\r\n    </div>\r\n    ");
}

<%#...%>以下是语法背后发生的事情的示例:

<script runat="server">
    public string SomeProperty { get { return "Hello World!"; } }
    protected void Page_Load(object sender, EventArgs e) { Label1.DataBind(); }
</script>

<form id="form1" runat="server">
    <div>
        <asp:Label ID="Label1" runat="server" Text="<%# SomeProperty %>"></asp:Label>
    </div>
</form>

生成此代码:

private Label @__BuildControlLabel1() 
{
    Label @__ctrl = new Label();
    
    this.Label1 = @__ctrl;
    @__ctrl.ApplyStyleSheetSkin(this);
    @__ctrl.ID = "Label1";
    @__ctrl.DataBinding += new System.EventHandler(this.@__DataBindingLabel1);
    return @__ctrl;
}

public void @__DataBindingLabel1(object sender, EventArgs e) 
{
    Label dataBindingExpressionBuilderTarget = ((Label)(sender));
    Page Container = ((Page)(dataBindingExpressionBuilderTarget.BindingContainer));
        
    dataBindingExpressionBuilderTarget.Text = System.Convert.ToString( SomeProperty , System.Globalization.CultureInfo.CurrentCulture);
}

如您所见,该<%=...%>语法可以在服务器控件的属性之外使用,以直接呈现返回值。另一方面,该<%#...%>语法为标签的 DataBinding 事件生成事件处理程序。此事件将标签属性的值设置为 SomeProperty 的值。每当调用 DataBind 方法时都会触发 DataBinding 事件,这就是我将该调用添加到 Page_Load 事件的原因。
希望这将帮助您了解它们之间的区别。

于 2009-08-14T22:13:54.637 回答
3

您可以创建一个自定义ExpressionBuilder以便使用类似的东西<%$ Code: SomeProperty %>

于 2009-08-13T10:38:15.513 回答
0

您可以创建自定义数据绑定控件,例如

namespace FooBar.WebControls
{
    public class DataBoundPlaceHolder:PlaceHolder
    {
        private bool hasDataBound = false;
        protected override void CreateChildControls()
        {
            if (!hasDataBound)
            {
                this.DataBind();
                hasDataBound = true;
            }
            base.CreateChildControls();
        }
    }
}

然后将您的代码包装在这个新控件中,并使用<%# %>语法,例如

<%@ Register TagPrefix="WebControls" Namespace="FooBar.WebControls" Assembly="FooBar" %>

<form id="form1" runat="server">
    <WebControls:DataBoundPlaceHolder runat="server">
        <asp:Label ID="Label1" runat="server" Text='<%# SomeProperty %>'></asp:Label>
    </WebControls:DataBoundPlaceHolder>
</form>
于 2014-04-23T10:02:07.450 回答