1

我一生都无法弄清楚 OnItemUpdated 何时被解雇。我一直在尝试学习 ASP.NET,所以你在这段代码中看到的一些东西可能是故意用困难的方式完成的(所以我可以更好地理解幕后发生的事情)

基本上,我有一个 GridView,它是使用 formview 作为细节的主控件。

这是SelectedIndexChanged方法GridView

protected void GridView1_SelectedIndexChanged(object sender, EventArgs e)
{
    var context = new DataAccessLayer.SafetyEntities();
    var se = (from c in context.Employees
                  where c.EID == (long)GridView1.SelectedDataKey.Value
                  select c).ToList();
    FormView1.DataSource = se;
    FormView1.DataKeyNames = new string[] { "EID" };
    FormView1.DataBind();
}

这很好,它在表单中显示选定的详细信息以进行编辑。这是看起来的formview样子:

<asp:FormView ID="FormView1" runat="server" DefaultMode="Edit" OnItemUpdating = "FormView1_ItemUpdating" OnItemUpdated="BLAH">
    <ItemTemplate>
        Select an employee!
     </ItemTemplate>
     <EditItemTemplate>
         <table>
             <tr>
                 <th>Name: 
                 </th>
                    <td>
                        <asp:TextBox runat="server" ID ="NameEdit" Text='<%#Bind("Name") %>' /> 
                    </td>
                    <br />
            </tr>
            <tr>
                <th>Manager: 
                </th>
                <td>
                    <asp:DropDownList ID = "DDLEdit1" DataSourceID = "ManagerEntitySource" runat="server"
                           DataTextField = "Option_Value" DataValueField = "Option_Value"
                           SelectedValue = '<%#Bind("Manager") %>'  
                           AppendDataBoundItems="true">
                    </asp:DropDownList> 
                </td>
                <br />
            </tr>
            <tr>
                <th>Location: 
                </th>
                <td>
                    <asp:DropDownList ID="DDLEdit2" DataSourceID = "LocationEntitySource" runat="server"
                           DataTextField = "Option_Value" DataValueField = "Option_Value"
                           SelectedValue='<%#Bind("Building") %>' 
                           AppendDataBoundItems="true">
                    </asp:DropDownList>
                </td>
                <br />
        </table>
        <asp:Button ID="Button2" Text="Submit Changes" runat="server" CommandName="Update" />
            <!--<asp:LinkButton ID = "LB1" Text="Update" CommandName="Update" runat="server" /> -->
    </EditItemTemplate>       
</asp:FormView>

这也有效。您可以从FormView我指定的OnItemUpdating和的属性中看到OnItemUpdated

这里是OnItemUpdating

protected void FormView1_ItemUpdating(object source, FormViewUpdateEventArgs e)
{
   DebugBox.Text = FormView1.DataKey.Value.ToString();
   DataAccessLayer.SafetyEntities se = new DataAccessLayer.SafetyEntities();
   var key = Convert.ToInt32(FormView1.DataKey.Value.ToString());
   DataAccessLayer.Employee employeeToUpdate = se.Employees.Where(emp => emp.EID == key).First();
   employeeToUpdate.Name = e.NewValues["Name"].ToString();
   employeeToUpdate.Manager = e.NewValues["Manager"].ToString();
   employeeToUpdate.Building = e.NewValues["Building"].ToString();
   se.SaveChanges();
   GridView1.DataBind();

}

这也很好。这些项目正在适当地更新并且GridView令人耳目一新。

这里是OnItemUpdated

protected void BLAH(object source, FormViewUpdatedEventArgs e)
{
    DebugBox2.Text = "BLAH!!!!";
}

这就是问题所在。这永远不会被调用!我是否在某个地方错过了让此事件触发的步骤?我以为我明白该按钮会调用 Command="Update",它会触发 ItemUpdating,然后触发 ItemUpdated。它肯定在调用 ItemUpdating,但仅此而已。我需要额外的东西来触发 ItemUpdated 吗?

4

1 回答 1

5

结论:

查看FormView该类的源代码,似乎ItemUpdated在使用 DataBinding 时未触发该事件。

仅当您设置、或的属性时ItemUpdated才会触发该事件。SelectMethodUpdateMethodDeleteMethodInsertMethodFormView

证据

当您在FormView其“UpdateItem”方法中更新一个项目时,该方法会在内部调用HandleUpdate.

public virtual void UpdateItem(bool causesValidation)
    {
      this.ResetModelValidationGroup(causesValidation, string.Empty);
      this.HandleUpdate(string.Empty, causesValidation);
    }

private void HandleUpdate(string commandArg, bool causesValidation)
    {
       // Lots of work is done here
    }

在方法的底部,触发HandleUpdate了事件:OnItemUpdating

this.OnItemUpdating(e);

然后是一段相当奇怪的代码:

if (e.Cancel || !bindingAutomatic)
          return;

依次调用Update.dataSourceView

dataSourceView.Update((IDictionary) e.Keys, (IDictionary) e.NewValues, 
(IDictionary) e.OldValues, 
new DataSourceViewOperationCallback(this.HandleUpdateCallback));

我们可以看到 Update 方法的最后一个参数接受了一个回调,在这种情况下是HandleUpdateCallback. HandleUpdateCallback是我们的OnItemUpdated事件最终被触发的地方。

private bool HandleUpdateCallback(int affectedRows, Exception ex)
    {
      FormViewUpdatedEventArgs e1 = new FormViewUpdatedEventArgs(
                                                       affectedRows, ex);
      e1.SetOldValues(this._updateOldValues);
      e1.SetNewValues(this._updateNewValues);
      e1.SetKeys(this._updateKeys);
      this.OnItemUpdated(e1);
      // A lot of other stuff goes on here
     }

所以,这就是我们最终如何到达OnItemUpdated正在执行的方法的地图,但为什么在我们的案例中它没有被执行?

让我们跳过一点,但回到HandleUpdate我之前提到的方法的“好奇”部分:

if (e.Cancel || !bindingAutomatic)
          return;

这里发生了什么?我们并没有取消我们的活动,但是那里!bindingAutomatic发生了什么?

该值在方法中进一步设置HandleUpdate

bool bindingAutomatic = this.IsDataBindingAutomatic;

这个属性 ,IsDataBindingAutomaticBaseDataBound类的一个内部属性(一个基类,比我们的FormView类更上一层):

  protected internal bool IsDataBindingAutomatic
    {
      get
      {
        if (!this.IsBoundUsingDataSourceID)
          return this.IsUsingModelBinders;
        else
          return true;
      }
    }

由于我们没有使用 aDataSourceID我们最终返回IsUsingModelBinders. 这是一个BaseDataBoundControl在类中被覆盖的虚拟属性CompositeDataBoundControl,类是我们FormView直接继承的基类。

现在我们来看看直接决定我们的OnItemUpdating方法是否会被触发的代码:

protected override bool IsUsingModelBinders
    {
      get
      {
        if (string.IsNullOrEmpty(this.SelectMethod) 
            && string.IsNullOrEmpty(this.UpdateMethod) 
            && string.IsNullOrEmpty(this.DeleteMethod))
          return !string.IsNullOrEmpty(this.InsertMethod);
        else
          return true;
      }
    }

这基本上是说如果我们设置了 SelectMethod、UpdateMethod 或 DeleteMethod(这些是 FormView 上的字符串属性),则返回 true,否则告诉我们是否设置了 InsertMethod。在我们的例子中,我们没有设置任何这些属性,所以我们得到一个返回值false.

因为这是错误的,我们之前两次奇怪的代码只是简单地返回并且永远不会到达触发 ItemUpdated 事件的代码部分。

于 2013-04-13T18:16:23.880 回答