主頁 > 知識庫 > 在ASP.NET 2.0中操作數(shù)據(jù)之十六:概述插入、更新和刪除數(shù)據(jù)

在ASP.NET 2.0中操作數(shù)據(jù)之十六:概述插入、更新和刪除數(shù)據(jù)

熱門標(biāo)簽:聯(lián)通官網(wǎng)400電話辦理 臨沂智能電話機(jī)器人加盟 西寧呼叫中心外呼系統(tǒng)線路商 地圖標(biāo)注軟件免費下載 400電話辦理怎么樣 網(wǎng)絡(luò)電話外呼系統(tǒng)上海 百應(yīng)電話機(jī)器人外呼系統(tǒng) 蘇州如何辦理400電話 外呼電話機(jī)器人成本

導(dǎo)言

  結(jié)束前面的幾節(jié),我們已經(jīng)探討過了如何使用GridView、DetailsView和FormView控件來顯示數(shù)據(jù)。這些控件簡單地操作提供給它的數(shù)據(jù)。一般地,這些控件通過使用一個數(shù)據(jù)源控件,例如ObjectDataSource來存取數(shù)據(jù)。我們已經(jīng)看過了ObjectDataSource是如何在ASP.NET頁面和潛在的數(shù)據(jù)之間扮演一個代理的角色。當(dāng)一個GridView需要顯示數(shù)據(jù)時,它調(diào)用ObjectDataSource的Select()方法,這個方法轉(zhuǎn)而調(diào)用一個來自我們的業(yè)務(wù)邏輯層(BLL)的方法,繼而調(diào)用一個適當(dāng)?shù)臄?shù)據(jù)訪問層(DAL)的表適配器(TableAdapter)的方法,從而它發(fā)送一個SELECT查詢到Northwind數(shù)據(jù)庫。

  記得在我們的教程里當(dāng)創(chuàng)建DAL中的表適配器時,Visual Studio自動地添加從潛在數(shù)據(jù)庫插入、更新和刪除數(shù)據(jù)的方法。此外,在創(chuàng)建一個業(yè)務(wù)邏輯層這一節(jié)我們已經(jīng)設(shè)計了調(diào)用這些數(shù)據(jù)更改的DAL方法的BLL方法。
除了它的Select()方法,ObjectDataSource還有Insert()、Update()和Delete()方法。跟Select()方法類似,這三個方法映射到一個隱含的對象。當(dāng)配置插入、更新或刪除數(shù)據(jù)時,GridView、DetailsView和FormView控件提供了一個修改潛在的數(shù)據(jù)的用戶界面。這個用戶界面調(diào)用ObjectDataSource的Insert()、Update()和Delete()方法,它們繼而調(diào)用隱含對象的關(guān)聯(lián)方法(見圖1)。

圖 1: ObjectDataSource的Insert()、Update()和Delete()方法提供一個到BLL的代理

  本節(jié)我們將看看如何映射ObjectDataSource的Insert()、Update()和Delete()方法到BLL中的類,也看看如何配置GridView、DetailsView和FormView控件提供修改數(shù)據(jù)的功能。

第一步: 創(chuàng)建Insert、Update和Delete教程頁面

  在我們開始探討如何插入、修改和刪除數(shù)據(jù)之前,讓我們先花些時間在我們的站點項目里添加這些本節(jié)里和下一節(jié)里需要的ASP.NET頁面。首先添加一個名為EditInsertDelete的新文件夾。然后,在這個文件夾里添加下面這些ASP.NET頁面,并且確認(rèn)每個頁面都關(guān)聯(lián)Site.master母版頁:

·Default.aspx
·Basics.aspx
·DataModificationEvents.aspx
·ErrorHandling.aspx
·UIValidation.aspx
·CustomizedUI.aspx
·OptimisticConcurrency.aspx
·ConfirmationOnDelete.aspx
·UserLevelAccess.aspx

圖 2: 添加這些與數(shù)據(jù)更改關(guān)聯(lián)的教程的頁面

  類似在其它文件夾里,EditInsertDelete文件夾里的Default.aspx將列出這些教程章節(jié)。記得用戶控件提供這個功能。因此,從解決方案資源管理器中拖拽一個這個用戶控件到頁面的設(shè)計視圖,從而添加它到Default.aspx頁面。

圖 3: 添加SectionLevelTutorialListing.ascx用戶控件到tDefault.aspx頁面

  最后,添加這些頁面地址項到Web.sitemap文件。明確地,在Customized Formatting siteMapNode>后添加如下標(biāo)記:

siteMapNode title="Editing, Inserting, and Deleting"
  url="~/EditInsertDelete/Default.aspx"
  description="Samples of Reports that Provide Editing, Inserting,
         and Deleting Capabilities">
  siteMapNode url="~/EditInsertDelete/Basics.aspx"
    title="Basics"
    description="Examines the basics of data modification with the
           GridView, DetailsView, and FormView controls." />
  siteMapNode url="~/EditInsertDelete/DataModificationEvents.aspx"
    title="Data Modification Events"
    description="Explores the events raised by the ObjectDataSource
           pertinent to data modification." />
  siteMapNode url="~/EditInsertDelete/ErrorHandling.aspx"
    title="Error Handling"
    description="Learn how to gracefully handle exceptions raised
           during the data modification workflow." />
  siteMapNode url="~/EditInsertDelete/UIValidation.aspx"
    title="Adding Data Entry Validation"
    description="Help prevent data entry errors by providing validation." />
  siteMapNode url="~/EditInsertDelete/CustomizedUI.aspx"
    title="Customize the User Interface"
    description="Customize the editing and inserting user interfaces." />
  siteMapNode url="~/EditInsertDelete/OptimisticConcurrency.aspx"
    title="Optimistic Concurrency"
    description="Learn how to help prevent simultaneous users from
           overwritting one another s changes." />
  siteMapNode url="~/EditInsertDelete/ConfirmationOnDelete.aspx"
    title="Confirm On Delete"
    description="Prompt a user for confirmation when deleting a record." />
  siteMapNode url="~/EditInsertDelete/UserLevelAccess.aspx"
    title="Limit Capabilities Based on User"
    description="Learn how to limit the data modification functionality
           based on the user role or permissions." />
/siteMapNode>

在更新了Web.sitemap后,花些時間通過瀏覽器訪問本教程站點。左邊的菜單里現(xiàn)在包含對應(yīng)編輯、插入和刪除教程的項。

圖 4: 站點地圖現(xiàn)在包含了對應(yīng)編輯、插入和刪除教程的項

第二步: 添加并配置ObjectDataSource控件

  因為GridView、DetailsView和FormView控件在數(shù)據(jù)修改功能和版面上都有所不同,就讓我們逐個研究。不過,與其讓這三個控件各自使用自己的ObjectDataSource,還不如讓我們僅創(chuàng)建一個ObjectDataSource讓這個三個控件的例子共用。
打開Basics.aspx頁面,從工具箱拖拽一個ObjectDataSource到設(shè)計器,從它的職能標(biāo)記中點擊配置數(shù)據(jù)源鏈接。因為ProductsBLL類是唯一一個提供修改、插入和刪除方法的BLL類,配置該ObjectDataSource使用這個類。

圖 5: 配置ObjectDataSource使用ProductsBLL類

  在下一屏中,通過選擇適當(dāng)?shù)膖ab頁并從下拉列表中選擇方法,我們可以指定ProductsBLL類里的哪些方法被映射到ObjectDataSource的Select()、Insert()、Update()和Delete()方法。圖6,至今我們應(yīng)該很熟悉,映射ObjectDataSource的Select()方法到ProductsBLL類的GetProducts()方法。Insert()、Update()和Delete()方法可以通過選擇上方的適當(dāng)?shù)膖ab頁進(jìn)行配置。

圖 6: 讓這個ObjectDataSource返回所有產(chǎn)品

  圖7、8和9顯示ObjectDataSource的UPDATE、INSERT,和DELETE 的tab頁。配置它們從而Insert()、Update()和Delete()方法分別調(diào)用ProductsBLL類的UpdateProduct、AddProduct和DeleteProduct方法。

圖 7: 映射ObjectDataSource的Update()方法到ProductBLL類的UpdateProduct方法

圖 8: 映射ObjectDataSource的Insert()方法到ProductBLL類的AddProduct方法

圖 9: 映射ObjectDataSource的Delete()方法到ProductBLL類的DeleteProduct方法

  你也許已經(jīng)注意到在UPDATE、INSERT和DELETE的tab頁里的下拉列表中已經(jīng)選擇了各自的方法。這是由于我們使用了DataObjectMethodAttribute,它修飾了ProducstBLL類。例如,DeleteProduct方法是如下這樣子聲明的:

[System.ComponentModel.DataObjectMethodAttribute
  (System.ComponentModel.DataObjectMethodType.Delete, true)]
public bool DeleteProduct(int productID)
{
  ...
}

DataObjectMethodAttribute指示每一個方法的目的–是否為了查詢、插入、更新或刪除–

是否它的默認(rèn)值。如果你在創(chuàng)建BLL類的時候省略了這些屬性,現(xiàn)在你將需要手工從UPDATE、INSERT和DELETE的tab頁里手工選擇方法。

當(dāng)確認(rèn)已經(jīng)適當(dāng)?shù)腜roductsBLL方法映射到ObjectDataSource的Insert()、Update()和Delete()方法后,點擊完成結(jié)束此向?qū)А?br /> 檢查ObjectDataSource的標(biāo)記

在通過數(shù)據(jù)源配置向?qū)瓿闪藢bjectDataSource的配置之后,到源視圖去檢查一下生成的聲明標(biāo)記。

asp:ObjectDataSource>標(biāo)簽列明了隱含的對象和需要調(diào)用的方法。另外,還有DeleteParameters、UpdateParameters和InsertParameters ,它們映射ProductsBLL類的AddProduct、UpdateProduct和DeleteProduct方法的輸入?yún)?shù):

asp:ObjectDataSource ID="ObjectDataSource1" runat="server"
  DeleteMethod="DeleteProduct" InsertMethod="AddProduct"
  OldValuesParameterFormatString="original_{0}" SelectMethod="GetProducts"
  TypeName="ProductsBLL" UpdateMethod="UpdateProduct">
  DeleteParameters>
    asp:Parameter Name="productID" Type="Int32" />
  /DeleteParameters>
  UpdateParameters>
    asp:Parameter Name="productName" Type="String" />
    asp:Parameter Name="supplierID" Type="Int32" />
    asp:Parameter Name="categoryID" Type="Int32" />
    asp:Parameter Name="quantityPerUnit" Type="String" />
    asp:Parameter Name="unitPrice" Type="Decimal" />
    asp:Parameter Name="unitsInStock" Type="Int16" />
    asp:Parameter Name="unitsOnOrder" Type="Int16" />
    asp:Parameter Name="reorderLevel" Type="Int16" />
    asp:Parameter Name="discontinued" Type="Boolean" />
    asp:Parameter Name="productID" Type="Int32" />
  /UpdateParameters>
  InsertParameters>
    asp:Parameter Name="productName" Type="String" />
    asp:Parameter Name="supplierID" Type="Int32" />
    asp:Parameter Name="categoryID" Type="Int32" />
    asp:Parameter Name="quantityPerUnit" Type="String" />
    asp:Parameter Name="unitPrice" Type="Decimal" />
    asp:Parameter Name="unitsInStock" Type="Int16" />
    asp:Parameter Name="unitsOnOrder" Type="Int16" />
    asp:Parameter Name="reorderLevel" Type="Int16" />
    asp:Parameter Name="discontinued" Type="Boolean" />
  /InsertParameters>
/asp:ObjectDataSource>

  ObjectDataSource包含了對應(yīng)它關(guān)聯(lián)的方法的每一個輸入?yún)?shù)的parameter,就像當(dāng)ObjectDataSource被配置為調(diào)用預(yù)期一個輸入?yún)?shù)的查詢方法(例如GetProductsByCategoryID(categoryID))時出現(xiàn)的SelectParameters一欄。正如我們馬上即將看到的,這些DeleteParameters、UpdateParameters和InsertParameters的值在調(diào)用ObjectDataSource的Insert()、Update()或Delete()方法之前自動地通過GridView、DetailsView和FormView被設(shè)置。必要時這些值也可以通過編程設(shè)置,這在以后的章節(jié)里討論。

  使用數(shù)據(jù)源配置向?qū)砼渲肙bjectDataSource的另一個影響是Visual Studio設(shè)置了OldValuesParameterFormatString屬性為original_{0}。這個屬性值用來包含數(shù)據(jù)被編輯時的原始值,它在下面兩種情況下非常有用:

·如果,當(dāng)編輯一條記錄時,用戶可以修改主鍵的值。在這種情況下,新的主鍵的值和原始的主鍵值都需要提供,這樣具有這個原始主鍵值的數(shù)據(jù)庫記錄才可以被找到然后才能將它的值更新。

·當(dāng)使用開放式并發(fā)。開放式并發(fā)是為了保證同時操作的用戶不至于覆蓋另一個用戶所做更改的一種技巧,這也是后面的教程中的一節(jié)(實現(xiàn)開放式并發(fā) )。

  這個OldValuesParameterFormatString屬性指明了隱含對象的更新和刪除方法中對應(yīng)原始值的輸入?yún)?shù)的名稱。我們將在探討開發(fā)式并發(fā)的時候更詳細(xì)地討論這個屬性和它的目的。不過暫時我放下它,因為我們的BLL的方法并不需要這些原始的值因此我們刪除這個屬性,這一點很重要。如果讓OldValuesParameterFormatString屬性設(shè)置為除了默認(rèn)值({0})以外的其它任何的值,都將在數(shù)據(jù)Web控件嘗試調(diào)用ObjectDataSource的Update()或Delete()方法時引發(fā)一個錯誤,因為ObjectDataSource將嘗試將這些原始值參數(shù)與UpdateParameters或DeleteParameters一起傳入。

  如果對此不是十分清楚,別擔(dān)心,我們將在未來的章節(jié)中研究這個屬性和它的效用。暫時,一定要完全地從聲明語法中完全地刪除這個屬性或者將它設(shè)置為默認(rèn)值({0})。

  注意: 如果你只是簡單地從設(shè)計視圖的屬性窗口刪除這個OldValuesParameterFormatString屬性的值,這個屬性依舊會存在于聲明語法中,不過被設(shè)置為一個空字符串。不幸地,這將依舊導(dǎo)致上面提到的同樣的問題。所以,從聲明語法里徹底地刪除這個屬性,或者從屬性窗口將其設(shè)置為默認(rèn)值,{0}。

第三步: 添加一個數(shù)據(jù)Web服務(wù)器控件并配置它為數(shù)據(jù)更改服務(wù)

  一般ObjectDataSOurce被添加到頁面并配置完成,我們可以添加一個數(shù)據(jù)Web服務(wù)器控件用來顯示數(shù)據(jù)并提供一個最終用戶修改數(shù)據(jù)的途徑。我們將分別看看GridView、DetailsView和FormView,因為這些數(shù)據(jù)Web服務(wù)器控件在它們的數(shù)據(jù)更改功能和配置上都有所不同。

  正如我們將在本文剩下的部分里看到的,通過GridView、DetailsView和FormView控件添加一個非常基本的編輯、插入和刪除支持是真的非常簡單,只需要勾選上一對CheckBox?,F(xiàn)實中提供這樣的功能有許多微妙之處和邊緣案例,這要比僅僅點幾下要棘手得多。但是,本教程里,只著眼于提供簡單的數(shù)據(jù)修改功能。以后的章節(jié)將研究在現(xiàn)實中不容置疑地出現(xiàn)的問題。

從GridView中刪除數(shù)據(jù)

  首先,從工具箱拖拽一個GridView到設(shè)計器。然后,通過GridView的智能標(biāo)記中從下拉列表中選擇從而綁定ObjectDataSource到該GridView。在這里GridView的聲明標(biāo)記將是:

asp:GridView ID="GridView1" runat="server" AutoGenerateColumns="False"
  DataKeyNames="ProductID" DataSourceID="ObjectDataSource1">
  Columns>
    asp:BoundField DataField="ProductID" HeaderText="ProductID"
      InsertVisible="False"
      ReadOnly="True" SortExpression="ProductID" />
    asp:BoundField DataField="ProductName" HeaderText="ProductName"
      SortExpression="ProductName" />
    asp:BoundField DataField="SupplierID" HeaderText="SupplierID"
      SortExpression="SupplierID" />
    asp:BoundField DataField="CategoryID" HeaderText="CategoryID"
      SortExpression="CategoryID" />
    asp:BoundField DataField="QuantityPerUnit"
      HeaderText="QuantityPerUnit"
      SortExpression="QuantityPerUnit" />
    asp:BoundField DataField="UnitPrice" HeaderText="UnitPrice"
      SortExpression="UnitPrice" />
    asp:BoundField DataField="UnitsInStock"
      HeaderText="UnitsInStock" SortExpression="UnitsInStock" />
    asp:BoundField DataField="UnitsOnOrder"
      HeaderText="UnitsOnOrder" SortExpression="UnitsOnOrder" />
    asp:BoundField DataField="ReorderLevel"
      HeaderText="ReorderLevel" SortExpression="ReorderLevel" />
    asp:CheckBoxField DataField="Discontinued"
      HeaderText="Discontinued" SortExpression="Discontinued" />
    asp:BoundField DataField="CategoryName"
      HeaderText="CategoryName" ReadOnly="True"
      SortExpression="CategoryName" />
    asp:BoundField DataField="SupplierName"
      HeaderText="SupplierName" ReadOnly="True"
      SortExpression="SupplierName" />
  /Columns>
/asp:GridView>

通過它的職能標(biāo)記綁定GridView到ObjectDataSource有下面兩點的好處:

·綁定列和CheckBox列被自動地添加,對應(yīng)ObjectDataSource返回的每一個字段。而且,這些綁定列和CheckBox列的屬性已經(jīng)被設(shè)置,基于隱含字段的元數(shù)據(jù)。例如ProductID、CategoryName和SupplierName列在ProductsDataTable里被標(biāo)記為只讀,因此它們在編輯時也是不可更新的。為了實現(xiàn)這一點,這些綁定列的ReadOnly屬性設(shè)置為true 。

·DataKeyNames屬性被賦值為隱含對象的主鍵。這是在使用GridView來編輯或刪除數(shù)據(jù)的要點,因為這個屬性象指出了標(biāo)識唯一記錄的那個字段(或是一組字段)。如果要獲得更多的關(guān)于DataKeyNames屬性的信息,請回到使用GridView 和DetailView實現(xiàn)的主/從報表一節(jié)。

雖然可以通過屬性窗口或者聲明語法將GridView綁定到ObjectDataSource,不過這需要你手工添加適當(dāng)?shù)慕壎泻虳ataKeyNames標(biāo)記。

GridView控件提供了對行編輯和刪除的內(nèi)建的支持。配置一個GridView支持刪除需要添加一個刪除按鈕列。當(dāng)最終用戶點擊某一特定行的刪除按鈕時,引發(fā)一次回傳并且GridView執(zhí)行以下步驟:

1.對ObjectDataSource的DeleteParameters賦值
2.調(diào)用ObjectDataSource的Delete()方法,刪除指定的記錄
3.通過調(diào)用它的Select()方法GridView重新綁定到ObjectDataSource

賦值到DeleteParameters的值是點擊刪除按鈕這一行的DataKeyNames字段的值。因此正確地設(shè)置GridView的DataKeyNames屬性是至關(guān)重要的。如果缺少了這個,DeleteParameters將在第1步被賦上一個null值,從而在第2步中將不會導(dǎo)致刪除任何記錄。

為了給GridView增加刪除功能,簡單地到它的職能標(biāo)記里勾選上“啟用刪除”。

圖 10: 勾選“啟用刪除”

從智能標(biāo)記中勾選啟用刪除會添加一個CommandField到GridView。這個CommandField在GridView中補(bǔ)充一個按鈕列,它履行一個或多個下屬任務(wù):選中一行記錄、編輯一行記錄和刪除一行記錄。我們先前在使用GridView 和DetailView實現(xiàn)的主/從報表一節(jié)的教程里也看到過了CommandField用作選中記錄時如何運作。

這個CommandFIeld包含了一些ShowXButton屬性,它指示哪一系列的按鈕顯示在CommandField中。通過勾選啟用刪除,一個ShowDeleteButton屬性為true的CommandField被添加到GridView的列集合。

asp:GridView ID="GridView1" runat="server" AutoGenerateColumns="False"
  DataKeyNames="ProductID" DataSourceID="ObjectDataSource1">
  Columns>
    asp:CommandField ShowDeleteButton="True" />
    ... BoundFields removed for brevity ...
  /Columns>
/asp:GridView>

到這里,你可能還不能相信,我們已經(jīng)完成了給這個GridView增加刪除支持!正如圖11顯示的,當(dāng)我們通過瀏覽器訪問此頁面時,一列刪除按鈕已經(jīng)出現(xiàn)。

圖 11: 這個CommandField添加一列刪除按鈕

  如果你是從一開始就是自己創(chuàng)建本教程的程序,當(dāng)測試這個頁面點擊刪除按鈕時將引發(fā)一個異常。繼續(xù)讀下去學(xué)習(xí)關(guān)于為什么會引發(fā)這些異常,還有如何修復(fù)它們。
  注意: 如果你是跟隨著下載的教程的程序,這些問題已經(jīng)被解決。然而,我鼓勵你從頭到尾讀一遍下面列出的詳細(xì)資料來幫助你識別可能出現(xiàn)的問題和適宜的工作區(qū)。
  如果,當(dāng)嘗試刪除一個產(chǎn)品,你得到一個類似于“ObjectDataSource 'ObjectDataSource1' could not find a non-generic method 'DeleteProduct' that has parameters: productID, original_ProductID,”的異常,你可能忘記了從ObjectDataSource里刪除OldValuesParameterFormatString屬性。指定了OldValuesParameterFormatString屬性的話,該ObjectDataSource會試圖向DeleteProduct方法一并傳入productID和original_ProductID輸入?yún)?shù),然而,DeleteProduct方法只能接受一個輸入?yún)?shù),導(dǎo)致異常。刪除OldValuesParameterFormatString屬性(或設(shè)置它為{0})指示ObjectDataSource不要試圖傳入這個原始值的輸入?yún)?shù)。

圖 12: 確保OldValuesParameterFormatString屬性已被徹底清除

  即使你已經(jīng)刪除了OldValuesParameterFormatString屬性,當(dāng)你嘗試刪除一個產(chǎn)品時依舊將得到一個異常:“The DELETE statement conflicted with the REFERENCE constraint 'FK_Order_Details_Products'.”。Northwind數(shù)據(jù)庫包含了一個在Order Details和Products表間的字段約束,表示如果一個產(chǎn)品在Order Details表里對應(yīng)它有一條或多條記錄,那么該產(chǎn)品不能被刪除。因為Northwind數(shù)據(jù)庫里的每一個產(chǎn)品在Order Details表里都至少有一條記錄,所以我們不能刪除任何產(chǎn)品,除非我們從order details表里刪除這個產(chǎn)品的關(guān)聯(lián)記錄。

圖 13: 一個字段間約束阻止了對產(chǎn)品的刪除

為了我們的教程,就讓我們刪除Order Details表里的所有記錄吧。在一個真實的應(yīng)用程序中我們需要的是下面任一措施:
·通過另外一個頁面管理order details信息
·在DeleteProduct方法里增加包含刪除指定產(chǎn)品的訂單明細(xì)的邏輯
·修改TableAdapter所使用的SQL語句,包含對指定產(chǎn)品的訂單明細(xì)的刪除
就讓我們從Order Details表里刪除所有記錄從而繞過字段間約束的問題。到Visual Studio的服務(wù)器資源管理器,在NORTHWND.MDF節(jié)點上點擊鼠標(biāo)右鍵,選擇“新建查詢”。然后,再查詢窗口執(zhí)行下面的SQL語句:DELETE FROM [Order Details]

圖 14: 從Order Details表里刪除所有記錄

  在清空了Order Details表后,點擊刪除按鈕將會正確無誤地刪除這個產(chǎn)品。如果點擊了刪除按鈕但是沒有刪除該產(chǎn)品,檢查并確保GridView的DataKeyNames屬性設(shè)置為主鍵(ProductID)。

  注意:當(dāng)點擊刪除按鈕時引發(fā)一次回傳并刪了了該記錄。這是危險的,因為它很容易意外地錯誤點擊了別的行的刪除按鈕。以后的章節(jié)里我們將看看如何在刪除記錄時添加一個客戶端的確認(rèn)詢問。

在GridView中編輯數(shù)據(jù)

  跟刪除支持一起,GridView還提供了內(nèi)建的對行編輯的支持。配置GrdiView支持編輯將添加一列編輯按鈕。從最終用戶的角度,點擊一行的編輯按鈕可使這一行變成可編輯的,它的單元格轉(zhuǎn)換成文本框并包含現(xiàn)有的值,并把編輯按鈕替換成保存和取消按鈕。在完成了他們期望的更改之后,最終用戶可以點擊保存按鈕提交這些修改,或者點擊取消按鈕放棄這些修改。在任意一種情況,點擊保存或者取消按鈕后GridView回到它編輯前的狀態(tài)。

站在我們頁面開發(fā)者的角度,當(dāng)最終用戶點擊特定一行的編輯按鈕時,引發(fā)一次回傳并且GridView執(zhí)行以下步驟:

1.GridView的EditItemIndex屬性被賦值為當(dāng)前點擊編輯按鈕的行的索引
2.通過調(diào)用它的Select()方法,GridView重新綁定自己到ObjectDataSource
3.與EditItemIndex相匹配的行呈現(xiàn)為編輯模式。在此模式下,編輯按鈕替換為保存和取消按鈕,并且那些ReadOnly屬性為False的綁定列呈現(xiàn)為TextBox服務(wù)器控件,這些TextBox的Text屬性被賦值為相應(yīng)的數(shù)據(jù)字段的值。

到這里HTML標(biāo)記被返回到瀏覽器,允許最終用戶可以修改行數(shù)據(jù)。當(dāng)用戶點擊保存按鈕,再次發(fā)生一次回傳,并且GridView執(zhí)

行以下幾個步驟:

1.ObjectDataSource的UpdateParameters的值被賦值為最終用戶在GridView的編輯界面輸入的值
2.調(diào)用ObjectDataSource的Update()方法,更新指定的記錄
3.通過調(diào)用它的Select()方法,GridView重新綁定自己到ObjectDataSource

  在DataKeyNames屬性指定的主鍵的值在第1步中賦值到UpdateParameters,反之非主鍵的值來自當(dāng)前編輯行的TextBox服務(wù)器控件。如果這一點遺漏了,那么UpdateParameters主鍵的值在第1步中將被賦上一個值,然后轉(zhuǎn)入第2步中將不會導(dǎo)致任何記錄的更新。

  編輯功能可以簡單地通過勾選GridView的智能標(biāo)記中的啟用編輯從而被激活。

圖 15: 勾選啟用編輯

  勾選啟用編輯將添加一個CommandField(如果需要的話)并設(shè)置它的ShowEditButton屬性為true 。如我們之前所看過的,CommandField包含一些ShowXButton屬性,他們指出哪一系列的按鈕要顯示在CommandField里。在我們的例子里,勾選啟用編輯添加ShowEditButton屬性到現(xiàn)有的CommandField里:

asp:GridView ID="GridView1" runat="server" AutoGenerateColumns="False"
  DataKeyNames="ProductID" DataSourceID="ObjectDataSource1">
  Columns>
    asp:CommandField ShowDeleteButton="True"
      ShowEditButton="True" />
    ... BoundFields removed for brevity ...
  /Columns>
/asp:GridView>

就這樣就添加了根本的編輯支持。如圖16所示,編輯界面還比較粗糙–每一個非只讀的綁定列呈現(xiàn)為一個TextBox。這包括像CategoryID和SupplierID的字段,而它們是來自其它表的外鍵。

圖 16: 點擊產(chǎn)品“Chai”的編輯按鈕讓此行顯示為“編輯模式”

為了要求用戶直接編輯外鍵,這個編輯界面還缺少了以下途徑:

·如果用戶輸入一個在數(shù)據(jù)庫中不存在的CategoryID或SupplierID,此更新將違反一個字段間的約束,引發(fā)一個異常。

·這個編輯界面不包含任何數(shù)據(jù)驗證。如果你不提供一個必填的值(例如ProductName),或者在要求輸入數(shù)字的地方輸入一個字符串,將拋出一個異常。未來的章節(jié)里將研究如何在編輯的用戶界面中增加驗證控件(給編輯和新增界面增加驗證控件)。

·通常,產(chǎn)品的所有非只讀字段都必須包含在GridView里。如果我們從GridView里剔除一列,比如說UnitPrice,當(dāng)更新數(shù)據(jù)時GridView將不會設(shè)置UnitPrice UpdateParameters的值,這將把該數(shù)據(jù)庫記錄的UnitPrice值更改為NULL值。類似地,如果一個必填的字段,例如ProductName,從GridView中被剔除了,那么這個更新將失敗,并出現(xiàn)一個上文中提及過的“Column 'ProductName' does not allow nulls”異常。

·這個編輯界面遺留了許多必要的格式化的問題。UnitPrice顯示為四位小數(shù)。理想地CategoryID和SupplierID應(yīng)該包含下拉列表(DropDownList),它列出系統(tǒng)中存在的類別和供應(yīng)商。

我們不得不承認(rèn)現(xiàn)在這還有許多缺點,但這些將在未來的章節(jié)里談及。

在DetailsView中插入、編輯和刪除數(shù)據(jù)

  正如我們在之前的章節(jié)里看過的,DetailsView控件一次只顯示一條記錄,就像GridView一樣,它也允許對當(dāng)前顯示的記錄進(jìn)行編輯和刪除。不管是對最終用戶來說從DetailsView進(jìn)行編輯和刪除的體驗,還是在ASP.NET這一面的工作流程,都跟GridView是一樣的。DetailsView和GridView不同的地方是,它還提供了內(nèi)鍵的插入支持。

  為了示范DetailsView的數(shù)據(jù)修改功能,首先,添加一個DetailsView控件到Basics.aspx頁面,放在現(xiàn)有的GridView的上方,并通過DetailsView的職能標(biāo)記把它綁定到現(xiàn)有的ObjectDataSource。然后,清除DetailsView的Height和Width屬性,并從它的職能標(biāo)記中勾選“啟用分頁”。為了啟用編輯、插入和刪除支持,只需要簡單地從它的職能標(biāo)記里勾選上“啟用插入”、“啟用編輯”和“啟用刪除”。

圖 17: 配置DetailsView支持編輯、插入和刪除

與GridView一樣,添加編輯、插入或刪除支持會添加一個CommandField到該DetailsView,如下聲明語法所示:

asp:DetailsView ID="DetailsView1" runat="server" AutoGenerateRows="False"
  DataKeyNames="ProductID" DataSourceID="ObjectDataSource1" AllowPaging="True">
  Fields>
    asp:BoundField DataField="ProductID"
      HeaderText="ProductID" InsertVisible="False"
      ReadOnly="True" SortExpression="ProductID" />
    asp:BoundField DataField="ProductName"
      HeaderText="ProductName" SortExpression="ProductName" />
    asp:BoundField DataField="SupplierID" HeaderText="SupplierID"
      SortExpression="SupplierID" />
    asp:BoundField DataField="CategoryID" HeaderText="CategoryID"
      SortExpression="CategoryID" />
    asp:BoundField DataField="QuantityPerUnit"
      HeaderText="QuantityPerUnit"
      SortExpression="QuantityPerUnit" />
    asp:BoundField DataField="UnitPrice"
      HeaderText="UnitPrice" SortExpression="UnitPrice" />
    asp:BoundField DataField="UnitsInStock"
      HeaderText="UnitsInStock" SortExpression="UnitsInStock" />
    asp:BoundField DataField="UnitsOnOrder"
      HeaderText="UnitsOnOrder" SortExpression="UnitsOnOrder" />
    asp:BoundField DataField="ReorderLevel"
      HeaderText="ReorderLevel" SortExpression="ReorderLevel" />
    asp:CheckBoxField DataField="Discontinued"
      HeaderText="Discontinued" SortExpression="Discontinued" />
    asp:BoundField DataField="CategoryName"
      HeaderText="CategoryName" ReadOnly="True"
      SortExpression="CategoryName" />
    asp:BoundField DataField="SupplierName"
      HeaderText="SupplierName" ReadOnly="True"
      SortExpression="SupplierName" />
    asp:CommandField ShowDeleteButton="True"
      ShowEditButton="True" ShowInsertButton="True" />
  /Fields>
/asp:DetailsView>

注意DetailsView的CommandField默認(rèn)顯示在列集合的下方。因為DetailsView的字段是以行的方式呈現(xiàn)的,所以CommandField也表現(xiàn)為包含插入、編輯和刪除按鈕的一行,出現(xiàn)在DetailsView的下方。

圖 18: 配置DetailsView支持編輯、插入和刪除

  點擊刪除按鈕就會開始與GridView相同的一系列的事件:一次回傳;隨之DetailsView基于DataKeyNames的值組成它的ObjectDataSOurce的DeleteParameters;最后以調(diào)用ObjectDataSource的Delete()方法結(jié)束,此方法從數(shù)據(jù)庫中刪除該產(chǎn)品的記錄。在DetailsView中編輯也以GridView同樣的方式運作。

  為了插入數(shù)據(jù),最終用戶面對的是一個“新建”按鈕,當(dāng)點擊時,DetailsView呈現(xiàn)為“插入模式”。在“插入模式”下,新建按鈕被“插入”和“取消”按鈕取代,并且顯示那些InsertVisible屬性設(shè)置為true(默認(rèn))的綁定列。這些自增長標(biāo)識的數(shù)據(jù)字段,例如ProductID字段,當(dāng)通過職能標(biāo)記綁定該DetailsView到數(shù)據(jù)源的時候讓它們的InsertVisible屬性設(shè)置為false 。

  當(dāng)通過智能標(biāo)記綁定數(shù)據(jù)源到DetailsView時,Visual Studio為自增長的字段設(shè)置其InsertVisible屬性為false。只讀的字段,像CategoryName和SupplierName,將顯示在“插入模式”下的用戶界面中,除非它們的InsertVisible屬性也明確地設(shè)置為false 。稍稍花些時間把這兩個字段的InsertVisible屬性設(shè)置為false ,通過DetailsView的聲明語法或者通過智能標(biāo)記中的“編輯字段”鏈接。

圖 19: Northwind商人現(xiàn)在提供產(chǎn)品“Acme Tea”

  在設(shè)置好InsertVisible屬性后,通過瀏覽器看看這個Basics.aspx頁面并點擊新建按鈕。圖20顯示的是添加一個新的飲料“Acme Tea”到我們的生產(chǎn)線時的DetailsView。

圖 20: Northwind商人現(xiàn)在提供產(chǎn)品“Acme Tea”

  輸入Acme Tea的詳細(xì)信息并點擊插入按鈕后,隨之發(fā)生一次回傳并將這個新記錄添加到Products數(shù)據(jù)表。因為這個DetailsView是按照數(shù)據(jù)庫中的順序依次列出產(chǎn)品,所以我們必須翻到最后一頁才能看到這個新增加的產(chǎn)品。

圖21: 產(chǎn)品Acme Tea的詳細(xì)信息

注意: DetailsView的CurrentMode 屬性指示當(dāng)前顯示的界面并可以被設(shè)置為下面幾個值之一:Edit、Insert或ReadOnly。DefaultMode屬性則指示DetailsView在完成一次編輯或插入之后顯示的模式,這在需要讓DetailsView保持編輯或插入模式不變時是很有用的。

  DetailsView這個點擊插入和編輯的功能跟GridView有相同的局限性:用戶必須通過文本框輸入存在的CategoryID和SupplierID值;界面缺少任何驗證的邏輯;產(chǎn)品的所有不允許為NULL值或者沒有在數(shù)據(jù)庫中指定默認(rèn)值的字段必須包含在插入界面里,等等。

  在以后的章節(jié)里我們將會研究的擴(kuò)展和提高GridView的編輯界面的技巧,同樣可以應(yīng)用到DetailsView的編輯和插入界面。

使用FormView做一個更靈活的數(shù)據(jù)修改用戶界面

  FormView控件提供內(nèi)建的對插入、編輯和刪除數(shù)據(jù)的支持,不過因為它使用模版而不是列,它沒有地方讓我們添加像GridView和DetailsView控件提供給數(shù)據(jù)修改界面的綁定列和CommandField。取而代之的是,這個界面 – 收集新增一項或編輯現(xiàn)有項時用來收集用戶輸入的Web服務(wù)器控件,連同新增、編輯、刪除、插入、保存和取消按鈕 – 都必須手工添加到適當(dāng)?shù)哪0胬?。幸運的是,Visual Studio將在通過它的職能標(biāo)記的下拉列表綁定FormView到數(shù)據(jù)源時自動地創(chuàng)建需要的界面。

  為了闡明這些技巧,首先,添加一個FormView控件到Basics.aspx頁面,并從FormView的職能標(biāo)記,綁定它到已經(jīng)存在的ObjectDataSource。這將為FormView生成一個EditItemTemplate、InsertItemTemplate和ItemTemplate ,用TextBox服務(wù)器控件收集用戶的輸入并用Button服務(wù)器控件作為添加新增、編輯、刪除、插入、保存和取消按鈕。另外,F(xiàn)ormView的DataKeyNames屬性被設(shè)置到ObjectDataSource所返回的對象的主鍵(ProductID)。最后,在FormView的職能標(biāo)記中勾選“起用分頁”選項。

  下面展示出FormView綁定到ObjectDataSource后它的ItemTemplate聲明標(biāo)記。默認(rèn)地,每一個除了布爾值以外的產(chǎn)品的字段都綁定到一個Label服務(wù)器控件的Text屬性,相應(yīng)地布爾類型的字段(Discontinued)綁定到一個不可更改的CheckBox服務(wù)器控件的Checked屬性。為了讓新增、編輯和刪除按鈕點擊時能夠引發(fā)某個FormView行為,必要的工作是將它們的CommandName屬性的值分別設(shè)置為New、Edit和Delete。

asp:FormView ID="FormView1" runat="server" DataKeyNames="ProductID"
  DataSourceID="ObjectDataSource1" AllowPaging="True">
  EditItemTemplate>
    ...
  /EditItemTemplate>
  InsertItemTemplate>
    ...
  /InsertItemTemplate>
  ItemTemplate>
    ProductID:
    asp:Label ID="ProductIDLabel" runat="server"
      Text='%# Eval("ProductID") %>'>/asp:Label>br />
    ProductName:
    asp:Label ID="ProductNameLabel" runat="server"
      Text='%# Bind("ProductName") %>'>
    /asp:Label>br />
    SupplierID:
    asp:Label ID="SupplierIDLabel" runat="server"
      Text='%# Bind("SupplierID") %>'>
    /asp:Label>br />
    CategoryID:
    asp:Label ID="CategoryIDLabel" runat="server"
      Text='%# Bind("CategoryID") %>'>
    /asp:Label>br />
    QuantityPerUnit:
    asp:Label ID="QuantityPerUnitLabel" runat="server"
      Text='%# Bind("QuantityPerUnit") %>'>
    /asp:Label>br />
    UnitPrice:
    asp:Label ID="UnitPriceLabel" runat="server"
      Text='%# Bind("UnitPrice") %>'>/asp:Label>br />
    UnitsInStock:
    asp:Label ID="UnitsInStockLabel" runat="server"
      Text='%# Bind("UnitsInStock") %>'>
    /asp:Label>br />
    UnitsOnOrder:
    asp:Label ID="UnitsOnOrderLabel" runat="server"
      Text='%# Bind("UnitsOnOrder") %>'>
    /asp:Label>br />
    ReorderLevel:
    asp:Label ID="ReorderLevelLabel" runat="server"
      Text='%# Bind("ReorderLevel") %>'>
    /asp:Label>br />
    Discontinued:
    asp:CheckBox ID="DiscontinuedCheckBox" runat="server"
      Checked='%# Bind("Discontinued") %>'
      Enabled="false" />br />
    CategoryName:
    asp:Label ID="CategoryNameLabel" runat="server"
      Text='%# Bind("CategoryName") %>'>
    /asp:Label>br />
    SupplierName:
    asp:Label ID="SupplierNameLabel" runat="server"
      Text='%# Bind("SupplierName") %>'>
    /asp:Label>br />
    asp:LinkButton ID="EditButton" runat="server"
      CausesValidation="False" CommandName="Edit"
      Text="Edit">
    /asp:LinkButton>
    asp:LinkButton ID="DeleteButton" runat="server"
      CausesValidation="False" CommandName="Delete"
      Text="Delete">
    /asp:LinkButton>
    asp:LinkButton ID="NewButton" runat="server"
      CausesValidation="False" CommandName="New"
      Text="New">
    /asp:LinkButton>
  /ItemTemplate>
/asp:FormView>

圖22顯示出通過瀏覽器查看時FormView的ItemTemplate。列出產(chǎn)品的每一個字段,并且在下方分別由新增、編輯和刪除按鈕。

圖 22:  FormView默認(rèn)的ItemTemplate列出產(chǎn)品的每一個字段并連同新增、編輯和刪除按鈕

  類似GridView和DetailsView,點擊刪除按鈕 – 或者其它任何一個CommandName屬性設(shè)置為“Delete”的Button、LinkButton或者ImageButton – 引發(fā)一次回傳,在FormView的DataKeyNames值的基礎(chǔ)上組建ObjectDataSource的DeleteParameters ,并調(diào)用ObjectDataSource的Delete()方法。

  當(dāng)點擊編輯按鈕時,引發(fā)一次回傳并且數(shù)據(jù)重新綁定到EditItemTemplate ,它作為開始的編輯界面。這個界面包含為編輯數(shù)據(jù)用的Web服務(wù)器控件連同保存和取消按鈕。默認(rèn)的通過Visual Studio生成的EditItemTemplate包含對應(yīng)自增長字段(ProductID)的Label、對應(yīng)每一個非布爾型字段的TextBox和對應(yīng)每一個布爾型字段的CheckBox 。這個動作與在GridView和DetailsView控件里自動生成的綁定列非常相似。

  注意: FormView的自動生成的EditItemTemplate有一個小問題,就是對于諸如CategoryName和SupplierName這樣的字段將TextBox服務(wù)器控件呈現(xiàn)為只讀。我們馬上看看如何解決這個問題。

  在EditItemTemplate里的TextBox控件的Text屬性已通過雙向綁定的方式綁定到相應(yīng)的數(shù)據(jù)字段的值。雙向綁定,以%# Bind("dataField") %>表示,在綁定數(shù)據(jù)到模版和為組裝ObjectDataSource插入或編輯記錄的參數(shù)時都回執(zhí)行綁定的動作。就是說,當(dāng)用戶從ItemTemplate里點擊編輯按鈕時,Bind()方法返回指定的數(shù)據(jù)字段的值。用戶完成他們的修改并點擊報存時,通過Bind()指定的數(shù)據(jù)字段的值回傳到ObjectDataSource的UpdateParameters 。作為另一種選擇,單向綁定,以%# Eval("dataField") %>表示,僅僅在綁定數(shù)據(jù)到模版時取得數(shù)據(jù)字段的值,但并不會在回傳時將用戶輸入的值返回到數(shù)據(jù)源控件的參數(shù)。

  下面的聲明標(biāo)記顯示了該FormView的EditItemTemplate。注意這里在綁定語法里用的是Bind()方法,并且因此保存和取消按鈕設(shè)置它們的CommandName屬性。 

asp:FormView ID="FormView1" runat="server" DataKeyNames="ProductID"
  DataSourceID="ObjectDataSource1" AllowPaging="True">
  EditItemTemplate>
    ProductID:
    asp:Label ID="ProductIDLabel1" runat="server"
     Text="%# Eval("ProductID") %>">/asp:Label>br />
    ProductName:
    asp:TextBox ID="ProductNameTextBox" runat="server"
     Text="%# Bind("ProductName") %>">
    /asp:TextBox>br />
    SupplierID:
    asp:TextBox ID="SupplierIDTextBox" runat="server"
     Text="%# Bind("SupplierID") %>">
    /asp:TextBox>br />
    CategoryID:
    asp:TextBox ID="CategoryIDTextBox" runat="server"
     Text="%# Bind("CategoryID") %>">
    /asp:TextBox>br />
    QuantityPerUnit:
    asp:TextBox ID="QuantityPerUnitTextBox" runat="server"
      Text="%# Bind("QuantityPerUnit") %>">
    /asp:TextBox>br />
    UnitPrice:
    asp:TextBox ID="UnitPriceTextBox" runat="server"
      Text="%# Bind("UnitPrice") %>">
    /asp:TextBox>br />
    UnitsInStock:
    asp:TextBox ID="UnitsInStockTextBox" runat="server"
      Text="%# Bind("UnitsInStock") %>">
    /asp:TextBox>br />
    UnitsOnOrder:
    asp:TextBox ID="UnitsOnOrderTextBox" runat="server"
      Text="%# Bind("UnitsOnOrder") %>">
    /asp:TextBox>br />
    ReorderLevel:
    asp:TextBox ID="ReorderLevelTextBox" runat="server"
      Text="%# Bind("ReorderLevel") %>">
    /asp:TextBox>br />
    Discontinued:
    asp:CheckBox ID="DiscontinuedCheckBox" runat="server"
      Checked="%# Bind("Discontinued") %>" />br />
    CategoryName:
    asp:TextBox ID="CategoryNameTextBox" runat="server"
       Text="%# Bind("CategoryName") %>">
    /asp:TextBox>br />
    SupplierName:
    asp:TextBox ID="SupplierNameTextBox" runat="server"
       Text="%# Bind("SupplierName") %>">
    /asp:TextBox>br />
    asp:LinkButton ID="UpdateButton" runat="server"
      CausesValidation="True" CommandName="Update"
      Text="Update">
    /asp:LinkButton>
    asp:LinkButton ID="UpdateCancelButton" runat="server"
      CausesValidation="False" CommandName="Cancel"
      Text="Cancel">
    /asp:LinkButton>
  /EditItemTemplate>
  InsertItemTemplate>
    ...
  /InsertItemTemplate>
  ItemTemplate>
    ...
  /ItemTemplate>
/asp:FormView>

  在這里我們的EditItemTemplate如果我們嘗試使用它的話將引起拋出一個異常。問題是CategoryName和SupplierName字段在EditItemTemplate里以TextBox服務(wù)器控件呈現(xiàn)。我們或者需要把這些TextBox替換為Label或者干脆把它們刪除。讓我們就簡單地從EditItemTemplate里把它們徹底刪除吧。

  圖23顯示的是在產(chǎn)品“Chai”上點擊編輯按鈕后瀏覽器中的FormView。注意顯示在ItemTemplate里的SupplierName和CategoryName字段現(xiàn)在不在出現(xiàn)了,因為我們剛剛從EditItemTemplate里刪除了它們。當(dāng)點擊保存按鈕時,F(xiàn)ormView執(zhí)行的是與GridView和DetailsView控件相似的一系列的步驟。

圖 23: EditItemTemplate默認(rèn)顯示每一個可編輯的產(chǎn)品字段為TextBox或CheckBox

  當(dāng)點擊FormView的ItemTemplate中的新建按鈕時發(fā)生一次回傳。不過,沒有數(shù)據(jù)綁定到FormView,因為新建了一條記錄。該InsertItemTemplate界面包含添加新記錄用的Web服務(wù)器控件連同插入和取消按鈕。通過Visual Studio生成的默認(rèn)的InsertItemTemplate包含對應(yīng)每一個非布爾型字段的TextBox和對應(yīng)布爾型字段的CheckBox,類似于自動生成的EditItemTemplate的界面。這些TextBox控件的Text屬性通過雙向綁定的方式綁定到相應(yīng)的數(shù)據(jù)字段。

  下面的聲明標(biāo)記顯示了該FormView的InsertItemTemplate。注意這里的綁定語法使用的是Bind()方法并且因此設(shè)置了插入和取消按鈕的CommandName屬性。

asp:FormView ID="FormView1" runat="server" DataKeyNames="ProductID"
  DataSourceID="ObjectDataSource1" AllowPaging="True">
  EditItemTemplate>
    ...
  /EditItemTemplate>
  InsertItemTemplate>
    ProductName:
    asp:TextBox ID="ProductNameTextBox" runat="server"
      Text="%# Bind("ProductName") %>">
    /asp:TextBox>br />
    SupplierID:
    asp:TextBox ID="SupplierIDTextBox" runat="server"
      Text="%# Bind("SupplierID") %>">
    /asp:TextBox>br />
    CategoryID:
    asp:TextBox ID="CategoryIDTextBox" runat="server"
      Text="%# Bind("CategoryID") %>">
    /asp:TextBox>br />
    QuantityPerUnit:
    asp:TextBox ID="QuantityPerUnitTextBox" runat="server"
      Text="%# Bind("QuantityPerUnit") %>">
    /asp:TextBox>br />
    UnitPrice:
    asp:TextBox ID="UnitPriceTextBox" runat="server"
      Text="%# Bind("UnitPrice") %>">
    /asp:TextBox>br />
    UnitsInStock:
    asp:TextBox ID="UnitsInStockTextBox" runat="server"
      Text="%# Bind("UnitsInStock") %>">
    /asp:TextBox>br />
    UnitsOnOrder:
    asp:TextBox ID="UnitsOnOrderTextBox" runat="server"
      Text="%# Bind("UnitsOnOrder") %>">
    /asp:TextBox>br />
    ReorderLevel:
    asp:TextBox ID="ReorderLevelTextBox" runat="server"
      Text="%# Bind("ReorderLevel") %>">
    /asp:TextBox>br />
    Discontinued:
    asp:CheckBox ID="DiscontinuedCheckBox" runat="server"
      Checked="%# Bind("Discontinued") %>" />br />
    CategoryName:
    asp:TextBox ID="CategoryNameTextBox" runat="server"
      Text="%# Bind("CategoryName") %>">
    /asp:TextBox>br />
    SupplierName:
    asp:TextBox ID="SupplierNameTextBox" runat="server"
      Text="%# Bind("SupplierName") %>">
    /asp:TextBox>br />
    asp:LinkButton ID="InsertButton" runat="server"
      CausesValidation="True" CommandName="Insert"
      Text="Insert">
    /asp:LinkButton>
    asp:LinkButton ID="InsertCancelButton" runat="server"
      CausesValidation="False" CommandName="Cancel"
      Text="Cancel">
    /asp:LinkButton>
  /InsertItemTemplate>
  ItemTemplate>
    ...
  /ItemTemplate>
/asp:FormView>

  FormView的自動生成的InsertItemTemplate有一點細(xì)微的區(qū)別。特別的是,對于只讀的字段,例如CategoryName和SupplierName,也為它們添加了相應(yīng)的TextBox服務(wù)器控件。類似EditItemTemplate,我們也需要從InsertItemTemplate里刪除這些TextBox。

  圖24顯示新增一個產(chǎn)品(Acme Coffee)時瀏覽器中的FormView。注意顯示在ItemTemplate中的SupplierName和CategoryName字段現(xiàn)在不再顯示了,因為我們剛剛刪除了它們。當(dāng)點擊插入按鈕時,F(xiàn)ormView執(zhí)行的是與GridView和DetailsView控件相似的一系列的步驟,插入一條新記錄到Products表。圖25顯示了新產(chǎn)品插入成功后在FormView中的詳細(xì)信息。

圖 24: InsertItemTemplate規(guī)定FormView的插入界面

圖 25: 新增產(chǎn)品Acme Coffee的詳細(xì)信息顯示在FormView中

通過分開read-only、editing和inserting界面到這3個不同的模版,F(xiàn)ormView可以比DetailsView和GridView更高程度地控制這些界面。

注意: 就像DetailsView,F(xiàn)ormView的CurrentMode屬性指示當(dāng)前顯示的界面,而它的DefaultMode屬性指示編輯或新建完成后FormView返回的顯示方式。

總結(jié)

在這一節(jié)里,我們研究了基本的使用GridView、DetailsView和FormView插入、編輯和刪除數(shù)據(jù)。這三種控件都提供一些內(nèi)建的數(shù)據(jù)修改功能,這可以被利用而不需要在ASP.NET頁面里寫一行代碼,這得益于數(shù)據(jù)Web控件和ObjectDataSource控件。不過,這個簡單的指和點的技巧只能提供出一個簡陋的數(shù)據(jù)修改用戶界面。為了提供數(shù)據(jù)驗證、注入編程設(shè)置的值、適當(dāng)?shù)靥幚懋惓?、自定義用戶界面等等,我們就需要依賴于一些在下面幾個章節(jié)了將討論的技巧。

祝編程快樂!

作者簡介

Scott Mitchell,著有六本ASP/ASP.NET方面的書,是4GuysFromRolla.com的創(chuàng)始人,自1998年以來一直應(yīng)用微軟Web技術(shù)。Scott是個獨立的技 術(shù)咨詢顧問,培訓(xùn)師,作家,最近完成了將由Sams出版社出版的新作,24小時內(nèi)精通ASP.NET 2.0。他的聯(lián)系電郵為mitchell@4guysfromrolla.com,也可以通過他的博客http://ScottOnWriting.NET與他聯(lián)系。

您可能感興趣的文章:
  • asp.net下gridview 批量刪除的實現(xiàn)方法
  • Asp.Net+XML操作基類(修改,刪除,新增,創(chuàng)建)
  • asp.net GridView 刪除時彈出確認(rèn)對話框(包括內(nèi)容提示)
  • asp.net中g(shù)ridview的查詢、分頁、編輯更新、刪除的實例代碼
  • asp.net 編輯gridview的小例子
  • Asp.net的GridView控件實現(xiàn)單元格可編輯方便用戶使用
  • ASP.NET MVC4入門教程(六):驗證編輯方法和編輯視圖
  • 在ASP.NET 2.0中操作數(shù)據(jù)之十七:研究插入、更新和刪除的關(guān)聯(lián)事件
  • 在ASP.NET 2.0中操作數(shù)據(jù)之十九:給編輯和新增界面增加驗證控件
  • 在ASP.NET 2.0中操作數(shù)據(jù)之二十二:為刪除數(shù)據(jù)添加客戶端確認(rèn)
  • 在ASP.NET 2.0中操作數(shù)據(jù)之三十六:在DataList里編輯和刪除數(shù)據(jù)概述

標(biāo)簽:聊城 清遠(yuǎn) 海西 慶陽 中衛(wèi) 臨夏 甘肅

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《在ASP.NET 2.0中操作數(shù)據(jù)之十六:概述插入、更新和刪除數(shù)據(jù)》,本文關(guān)鍵詞  在,ASP.NET,2.0,中,操作,數(shù)據(jù),;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《在ASP.NET 2.0中操作數(shù)據(jù)之十六:概述插入、更新和刪除數(shù)據(jù)》相關(guān)的同類信息!
  • 本頁收集關(guān)于在ASP.NET 2.0中操作數(shù)據(jù)之十六:概述插入、更新和刪除數(shù)據(jù)的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章