【问题标题】:Bind data repeater inside data repeater in ASP.NET在 ASP.NET 中的数据中继器内绑定数据中继器
【发布时间】:2012-06-05 14:18:38
【问题描述】:

我现在正在学习 .NET,如果这是一个愚蠢的问题,请原谅我。

我有两个 MSSQL 表,一个名为“Comment”,另一个名为“CommentAdditionalAuthor”。 Comment 表包含来自评论表单的数据,例如评论标题、日期、主要作者姓名等。CommentAdditionalAuthor 表包含有关对评论做出贡献的其他作者的信息。所以那里是一对多的关系——每条评论可以有一个或多个额外的作者。

我有一个转发器控件,可以绑定来自 Comment 表的数据,效果很好。现在我想做的是在评论旁边输出作者的详细信息。如此有效的中继器中的中继器。

我已尝试尽可能多地清理此代码,但很抱歉它仍然很长,因为我想让它与我正在做的事情保持相关,以便我希望能够理解解决方案。找到其他示例后,我正在努力实现它们,因为它们以不同的方式绑定到数据,而我无法在我的应用程序中使用。

aspx文件中的代码是这样的:

<div id="comments">
    <asp:Repeater ID="rptComments" runat="server">
    <ItemTemplate>
        <div class="comment-data">
            <h3 class="item">Submitted to <%# GetPageDetails(Eval("nodeid")) %> article on <%# Eval("created") %></strong></h3>
            <p class="item"><strong>Name</strong> <%# Eval("firstname") %> <%# Eval("surname") %></p>
            <p class="item"><strong>Occupation</strong> <%# Eval("occupation") %></p>
            <p class="item"><strong>Affiliation</strong> <%# Eval("affiliation") %></p>
            <p class="item"><strong>Email</strong> <a href='mailto:<%# Eval("email") %>'><%# Eval("email") %></a> <em>Publish email: <%# Eval("publishemail") %></em></p>
            <p class="item"><strong>Competing interests?</strong> <%# Eval("competingintereststext") %>&nbsp;</p>
            <p class="item"><strong>eLetter title</strong> <%# Eval("title") %></p>
            <p><%# Eval("comment").ToString().Replace("\n", "<br/>")%></p>


            <!-- This is what I want to do, but can't get it to bind: -->
            <div class="additional-authors">
                <h3>Additional authors</h3>
                <asp:Repeater id="rptAdditionalAuthors" runat="server">
                    <ItemTemplate>
                        <%# Eval("firstnameother")%><br>
                        <%# Eval("surnameother")%><br>
                        <%# Eval("occupationother")%><br>
                        <%# Eval("affiliationother")%><br>
                        <%# Eval("emailother")%><br>
                    </ItemTemplate>
                </asp:Repeater>
            </div>

        </div>                
    </ItemTemplate>
</asp:Repeater>

下面是代码隐藏的作用:

    private void BindData()
    {
        var rr = _sqlHelper.ExecuteReader(string.Format("select * from comment {0} order by created desc", Filter));

        var commentDt = new DataTable("Comments");
        commentDt.Columns.Add("id", Type.GetType("System.Int32"));
        commentDt.Columns.Add("nodeid", Type.GetType("System.Int32"));
        commentDt.Columns.Add("firstname", Type.GetType("System.String"));
        commentDt.Columns.Add("surname", Type.GetType("System.String"));
        commentDt.Columns.Add("occupation", Type.GetType("System.String"));
        commentDt.Columns.Add("affiliation", Type.GetType("System.String"));
        commentDt.Columns.Add("title", Type.GetType("System.String"));
        commentDt.Columns.Add("email", Type.GetType("System.String"));
        commentDt.Columns.Add("publishemail", Type.GetType("System.Boolean"));
        commentDt.Columns.Add("competinginterests", Type.GetType("System.Boolean"));
        commentDt.Columns.Add("competingintereststext", Type.GetType("System.String"));
        commentDt.Columns.Add("comment", Type.GetType("System.String"));
        commentDt.Columns.Add("statusid", Type.GetType("System.Int32"));
        commentDt.Columns.Add("spam", Type.GetType("System.Boolean"));
        commentDt.Columns.Add("ham", Type.GetType("System.Boolean"));
        commentDt.Columns.Add("created",Type.GetType("System.DateTime"));

        while (rr.Read())
        {
            var dr = commentDt.NewRow();
            dr["id"] = rr.GetInt("id");
            dr["nodeid"] = rr.GetInt("nodeid");
            dr["firstname"] = rr.GetString("firstname");
            dr["surname"] = rr.GetString("surname");
            dr["occupation"] = rr.GetString("occupation");
            dr["affiliation"] = rr.GetString("affiliation");
            dr["title"] = rr.GetString("title");
            dr["email"] = rr.GetString("email");
            dr["publishemail"] = rr.IsNull("publishemail") == true ? false : rr.GetBoolean("publishemail");
            dr["competinginterests"] = rr.IsNull("competinginterests") == true ? false : rr.GetBoolean("competinginterests");
            dr["competingintereststext"] = rr.GetString("competingintereststext");
            dr["comment"] = rr.GetString("comment");
            dr["statusid"] = rr.GetInt("statusid");
            dr["spam"] = rr.IsNull("spam") == true ? false : rr.GetBoolean("spam");
            dr["ham"] = rr.IsNull("ham") == true ? false : rr.GetBoolean("ham");
            dr["created"] = rr.GetDateTime("created");

            commentDt.Rows.Add(dr);


            /* My failing attempt at the second bind:
            var rrAuthor = _sqlHelper.ExecuteReader(string.Format("select * from CommentOtherAuthor WHERE commentid = 81"));

            var otherAuthorDt = new DataTable("OtherAuthors");
            otherAuthorDt.Columns.Add("firstname", Type.GetType("System.String"));
            otherAuthorDt.Columns.Add("surname", Type.GetType("System.String"));
            otherAuthorDt.Columns.Add("occupation", Type.GetType("System.String"));
            otherAuthorDt.Columns.Add("affiliation", Type.GetType("System.String"));
            otherAuthorDt.Columns.Add("email", Type.GetType("System.String"));

            while (rrAuthor.Read())
            {
                var drAuthor = otherAuthorDt.NewRow();
                drAuthor["firstnameother"] = rr.GetString("firstname");
                drAuthor["surnameother"] = rr.GetString("surname");
                drAuthor["occupationother"] = rr.GetString("occupation");
                drAuthor["affiliationother"] = rr.GetString("affiliation");
                drAuthor["emailother"] = rr.GetString("email");
                otherAuthorDt.Rows.Add(drAuthor);
            }
            rptAdditionalAuthors.DataBind();
            */
        }

        var pgitems = new PagedDataSource
            {
                DataSource = commentDt.DefaultView,
                AllowPaging = true,
                PageSize = 25,
                CurrentPageIndex = CurrentPage
            };

        rptComments.DataSource = pgitems;
        rptComments.DataBind();

        if (pgitems.PageCount > 1)
        {

            var pages = new ArrayList();
            for (var i = 0; i < pgitems.PageCount; i++)
                pages.Add((i + 1).ToString());

            rptPages.DataSource = pages;
            rptPages.DataBind();
            rptPages.Visible = true;
        }
        else
        {
            rptPages.Visible = false;

        }
    }

由于我设置了第二个中继器,代码隐藏的东西不起作用,但如果我删除所有这些东西,第一个中继器确实起作用。我认为最好将其保留在那里,这样您就可以看到我的“逻辑”(可能不是很合乎逻辑;))。另外值得注意的是,在第二个数据集中,我对其进行了硬编码,以获取 id 为 81 的其他作者详细信息——这只是为了让中继器工作——显然我需要用 Comment 表中的 id 值替换它,在循环中获取当前评论的作者。

如果有人能帮助我让第二个中继器正常工作,我将不胜感激。目前它已经超出了我的学习曲线的边缘!

谢谢各位!

【问题讨论】:

    标签: asp.net data-binding repeater nested-repeater


    【解决方案1】:

    有几篇关于嵌套中继器的在线文章,包括this one

    本质上,您需要首先创建您的数据集并在代码隐藏中定义数据表之间的关系,然后将子转发器绑定到子行。上面链接的文章有一个完整的工作示例,很容易修改。

    如果该文章不适合您的需求,请尝试one of these

    【讨论】:

    • 谢谢。第一篇文章是我一直试图关注的,但我无法让它工作——它似乎使用了一堆新的东西,比如我的应用程序中没有的 SQLDataAdapter,我不太理解足以知道每个部分在做什么以及在我的示例中等效的是什么。不过,我会看看你的第二个链接。
    • DataAdatper 用于填充 DataSet,从而创建数据的内存版本。您正在使用 SqlDatareader,它允许您一次遍历一条记录。它们是获取数据的两种不同方法。要进行嵌套中继器,最简单的方法是使用 DataAdapters。稍等一下,我会为您找到一篇概述差异的快速文章。切换应该不会太难。
    • 从 DataAdapter 填充数据集:msdn.microsoft.com/en-us/library/bh8kx08z.aspx
    • 关于 dataAdapter 的详细文章:developerfusion.com/article/84366/adonet-dataadapter 无论如何,使用 dataAdapter 是一项基本的 .NET 技能,因此了解它们很重要。祝你好运!
    • 我从来没有遇到过允许你这样做的代码。我看到的唯一接近您所要求的事情是不使用中继器,而是使用 StringBuilders 手动构建 HTML。据我所知,真正的 DataBinding 要在嵌套场景中工作,您绝对需要定义表之间的关系,这需要 DataSet。您可以通过手动构建表、循环读取器的结果集并向表中添加行来使用 DataReader 填充 DataSet,但这种方法很麻烦,不推荐使用。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多