【问题标题】:Why is the DataItem null for a GridView row "null" when trying to save changes?为什么在尝试保存更改时 GridView 行的 DataItem 为空?
【发布时间】:2012-11-08 23:16:27
【问题描述】:

所以我的网页上有这个 GridView。它是数据绑定的,因此在 RowDataBound 事件期间,这段代码运行良好:

    protected void myGridView_RowDataBound(object sender, GridViewRowEventArgs e)
    {
        if (e.Row.RowType == DataControlRowType.DataRow)
        {
            TimecardApproval shift = (TimecardApproval)e.Row.DataItem;
        }
    }

“shift”变量包含我需要的所有数据,效果很好。但是还有这段代码用于保存数据:

    protected void btnSubmitApprovals_Click(object sender, ImageClickEventArgs e)
    {
        foreach (GridViewRow item in gvTimeCards.Rows)
        {
            TimecardApproval shift = (TimecardApproval)item.DataItem;
            // code to update the row
        }
    }

DataItem 为空!为什么?由于该字段在那里,它为空似乎很奇怪。我应该循环其他东西吗?

【问题讨论】:

  • DataItem 在数据绑定期间设置。在回发时,它始终为空,直到您再次DataBind GridView。请注意,所有变量都在页面生命周期结束时处理。 btnSubmitApprovals_Click 你到底需要什么?通常,您可以从 Bound-/TemplateFields 中的控件获取它。
  • @TimSchmelter 文档说:DataItem 属性仅在 GridView 控件的 RowDataBound 事件期间和之后可用。你能澄清一下“之后”部分吗?
  • @codingbiz: After 表示在同一页面生命周期中数据绑定之后(所以直到Unload)。
  • @TimSchmelter,获取原始 DataItem 似乎是找出哪些字段已被修改的最合乎逻辑的方法。那你是说我可以在click函数中重新绑定GridView?
  • @TheGerm:不,您不能重新绑定 GridView。然后,您将覆盖所有更改并防止引发事件。您需要评估控件或使用 GridView RowUpdating 事件及其 OldValuesNewValues 属性。

标签: asp.net gridview


【解决方案1】:

我了解您可能希望通过将 DataItem 转换为 TimeCardApproval 来获得智能感知。您可以改为这样做。使用 DataKeyNames 存储应为 TimecardApprovalID 的每一行的主键,并在您的代码中访问主键并使用它来获取原始项目

foreach (GridViewRow item in gvTimeCards.Rows)
{
   //get the ID of the TimeApproval for each row
   string id = gvDocs.DataKeys[item.RowIndex].Value.ToString();
   //string id = ((HiddenField) item.FindControl("IDHiddenField")).Value;
   //string id = item.Cells[0].Text;

   //use the ID or get TimeCardApproval object from DB    
   TimecardApproval shift = MyDB.GetTimeCardApproval(id);

}

设置键

<asp:GridView ID="gvTimeCards" DataKeyNames="TimecardApprovalID">
</asp:GridView>

【讨论】:

  • 这对我来说就像一个冠军,尽管我认为更好的方法是使用 RowUpdating 操作。此外,我在转换 DataItem 时没有问题,但我不知道如何获取 DataItem 或如何获取参考来查找它。此 DataKey 信息非常有用。谢谢!您能否编辑您的答案以说“这是如何获得对原始项目的引用”,以便我可以接受这个答案?
  • 只是一个后续。我实际上必须走这条路,因为 GridView 正在使用的数据源没有“更新”功能。 ASP.NET 的速成课程。
  • 发现这更有帮助:stackoverflow.com/questions/4633261/…
  • 性能怎么样?每行访问数据库(使用 ID)? 30 行,30 次调用数据库?
  • @Kiquenet 你可以缓存你的数据或使用内存中的集合
【解决方案2】:

也许,这对你有帮助
网格列:

<asp:TemplateField HeaderText="Item ID">
<ItemTemplate>
    <asp:Label runat="server" Text='<%# DataBinder.Eval(Container.DataItem, "Id") %>'/>
    <asp:HiddenField runat="server" ID="hfDataItem" Value='<%# JsonConvert.SerializeObject(Container.DataItem) %>'/>
</ItemTemplate></asp:TemplateField>

后面的代码:

    protected void btApply_OnClick(object sender, EventArgs e)
    {
        var rows = (from GridViewRow row in gvResult.Rows
            where row.RowType == DataControlRowType.DataRow
            let dataItem = (HiddenField) row.FindControl("hfDataItem")
            select JsonConvert.DeserializeObject<ResultGridRowItem>(dataItem.Value));

    }

【讨论】:

    【解决方案3】:

    gridview 数据绑定事件在按钮单击之前触发。听起来很奇怪,但那是asp.net page life cycle 的疯狂。 尝试在 gridview RowUpdatingRowUpdated 事件中使用您的代码。

    【讨论】:

    • 在知道原始值是什么之前,我不知道是否要更新该行。但也许你是说每一行都应该处理自己的更新?如果我愿意,我可以“取消”行更新吗?
    • 是的,e.Cancel 就是您所说的。
    猜你喜欢
    • 1970-01-01
    • 2017-04-13
    • 1970-01-01
    • 2011-01-23
    • 2023-04-07
    • 1970-01-01
    • 2020-06-02
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多