Condividi tramite


Implementazione della concorrenza ottimistica (C#)

di Scott Mitchell

Scarica il PDF

Per un'applicazione Web che consente a più utenti di modificare i dati, c'è il rischio che due utenti possano modificare contemporaneamente gli stessi dati. In questa esercitazione verrà implementato il controllo di concorrenza ottimistica per gestire questo rischio.

Introduzione

Per le applicazioni Web che consentono solo agli utenti di visualizzare i dati o per quelli che includono solo un singolo utente che può modificare i dati, non esiste alcuna minaccia di due utenti simultanei sovrascrivendo accidentalmente le modifiche di un altro. Per le applicazioni Web che consentono a più utenti di aggiornare o eliminare i dati, tuttavia, è possibile che le modifiche di un utente siano in contrasto con un altro utente simultaneo. Senza alcun criterio di concorrenza, quando due utenti modificano simultaneamente un singolo record, l'utente che esegue il commit delle modifiche l'ultima eseguirà l'override delle modifiche apportate dal primo.

Si supponga, ad esempio, che due utenti, Jisun e Sam, visitassero entrambe una pagina dell'applicazione che consente ai visitatori di aggiornare ed eliminare i prodotti tramite un controllo GridView. Entrambi fare clic sul pulsante Modifica in GridView contemporaneamente. Jisun modifica il nome del prodotto in "Chai Tea" e fa clic sul pulsante Aggiorna. Il risultato netto è un'istruzione UPDATE inviata al database, che imposta tutti i campi aggiornabili del prodotto (anche se Jisun ha aggiornato un solo campo, ProductName). In questo momento, il database ha i valori "Chai Tea", la categoria Bevande, il fornitore Esotico Liquids e così via per questo particolare prodotto. Tuttavia, gridView nella schermata di Sam mostra ancora il nome del prodotto nella riga GridView modificabile come "Chai". Alcuni secondi dopo il commit delle modifiche di Jisun, Sam aggiorna la categoria nei condimenti e fa clic su Aggiorna. Ciò comporta un'istruzione UPDATE inviata al database che imposta il nome del prodotto su "Chai", sull'ID CategoryID categoria Bevande corrispondente e così via. Le modifiche apportate a Jisun al nome del prodotto sono state sovrascritte. La figura 1 illustra graficamente questa serie di eventi.

Quando due utenti aggiornano simultaneamente un record c'è potenziale per le modifiche apportate a un utente per sovrascrivere l'altro

Figura 1: Quando due utenti aggiornano simultaneamente un record, è possibile che un utente cambi per sovrascrivere l'altro (fare clic per visualizzare l'immagine full-size)

Analogamente, quando due utenti visitano una pagina, un utente potrebbe trovarsi nel corso dell'aggiornamento di un record quando viene eliminato da un altro utente. In alternativa, tra quando un utente carica una pagina e quando fa clic sul pulsante Elimina, un altro utente potrebbe aver modificato il contenuto di tale record.

Sono disponibili tre strategie di controllo concorrenza :

  • Do Nothing -if gli utenti simultanei stanno modificando lo stesso record, lasciare che l'ultimo commit win (il comportamento predefinito)
  • Concorrenza ottimistica : si supponga che, mentre ci saranno conflitti di concorrenza ogni ora e poi, la maggior parte del tempo tali conflitti non si verificheranno; pertanto, se si verifica un conflitto, informa semplicemente l'utente che le modifiche non possono essere salvate perché un altro utente ha modificato gli stessi dati
  • Concorrenza pessimistica : si supponga che i conflitti di concorrenza siano comuni e che gli utenti non tollerano che le modifiche non siano state salvate a causa dell'attività simultanea di un altro utente; pertanto, quando un utente avvia l'aggiornamento di un record, bloccarlo, impedendo così ad altri utenti di modificare o eliminare tale record fino a quando l'utente esegue il commit delle modifiche

Tutte le esercitazioni finora hanno usato la strategia di risoluzione della concorrenza predefinita, ovvero l'ultima vittoria di scrittura. In questa esercitazione verrà illustrato come implementare il controllo di concorrenza ottimistica.

Nota

In questa serie di esercitazioni non verranno esaminati esempi di concorrenza pessimistici. La concorrenza pessimistica viene usata raramente perché tali blocchi, se non vengono riscriuti correttamente, possono impedire ad altri utenti di aggiornare i dati. Ad esempio, se un utente blocca un record per la modifica e quindi lascia per il giorno prima di sbloccarlo, nessun altro utente sarà in grado di aggiornare tale record fino a quando l'utente originale non restituisce e completa l'aggiornamento. Pertanto, in situazioni in cui viene usata la concorrenza pessimistica, si verifica in genere un timeout che, se raggiunto, annulla il blocco. I siti Web di vendita dei ticket, che bloccano una determinata posizione a sedere per breve periodo mentre l'utente completa il processo di ordine, è un esempio di controllo di concorrenza pessimistico.

Passaggio 1: Esaminare come viene implementata la concorrenza ottimistica

Il controllo di concorrenza ottimistica funziona assicurandosi che il record aggiornato o eliminato abbia gli stessi valori di quando è stato avviato l'aggiornamento o l'eliminazione del processo. Ad esempio, quando si fa clic sul pulsante Modifica in un controllo GridView modificabile, i valori del record vengono letti dal database e visualizzati in Caselle di testo e in altri controlli Web. Questi valori originali vengono salvati da GridView. Successivamente, dopo che l'utente apporta le modifiche e fa clic sul pulsante Aggiorna, i valori originali e i nuovi valori vengono inviati al livello di logica di business e quindi al livello di accesso ai dati. Il livello di accesso ai dati deve emettere un'istruzione SQL che aggiornerà solo il record se i valori originali che l'utente ha avviato la modifica sono identici ai valori ancora presenti nel database. La figura 2 illustra questa sequenza di eventi.

Per eseguire l'aggiornamento o l'eliminazione, i valori originali devono essere uguali ai valori correnti del database

Figura 2: Per l'aggiornamento o l'eliminazione, i valori originali devono essere uguali ai valori del database corrente (fare clic per visualizzare l'immagine a dimensioni complete)

Esistono vari approcci all'implementazione della concorrenza ottimistica (vedere Peter A. Bromberg'sOptimistic Concurrency Update Logic per una breve analisi di una serie di opzioni). Il ADO.NET Typed DataSet fornisce un'implementazione che può essere configurata solo con il segno di spunta di una casella di controllo. L'abilitazione UPDATE della concorrenza ottimistica per un TableAdapter nell'oggetto Typed DataSet aumenta le istruzioni e DELETE di TableAdapter per includere un confronto di tutti i valori originali nella WHERE clausola . L'istruzione seguente UPDATE , ad esempio, aggiorna il nome e il prezzo di un prodotto solo se i valori correnti del database sono uguali ai valori recuperati originariamente durante l'aggiornamento del record in GridView. I @ProductName parametri e @UnitPrice contengono i nuovi valori immessi dall'utente, mentre @original_ProductName e @original_UnitPrice contengono i valori originariamente caricati in GridView quando è stato fatto clic sul pulsante Modifica:

UPDATE Products SET
    ProductName = @ProductName,
    UnitPrice = @UnitPrice
WHERE
    ProductID = @original_ProductID AND
    ProductName = @original_ProductName AND
    UnitPrice = @original_UnitPrice

Nota

Questa UPDATE istruzione è stata semplificata per la leggibilità. In pratica, il UnitPrice controllo nella WHERE clausola sarebbe più coinvolto poiché UnitPrice può contenere NULL s e controllare se NULL = NULL restituisce sempre False (invece è necessario usare IS NULL).

Oltre a usare un'istruzione sottostante UPDATE diversa, la configurazione di un TableAdapter per l'uso della concorrenza ottimistica modifica anche la firma dei metodi diretti del database. Richiamare dalla prima esercitazione la creazione di un livello di accesso ai dati, ovvero i metodi diretti del database che accettano un elenco di valori scalari come parametri di input (anziché un'istanza di DataRow o DataTable fortemente tipizzata). Quando si usa la concorrenza ottimistica, i metodi diretti Update() e Delete() del database includono anche i parametri di input per i valori originali. Inoltre, il codice in BLL per l'uso del modello di aggiornamento batch (gli Update() overload del metodo che accettano DataRows e DataTables anziché i valori scalari) devono essere modificati anche.

Anziché estendere i TableAdapter esistenti di DAL per usare la concorrenza ottimistica (che richiederebbe la modifica del BLL in modo da adattare), verrà invece creato un nuovo Set di dati tipizzato denominato NorthwindOptimisticConcurrency, a cui si aggiungerà un Products TableAdapter che usa la concorrenza ottimistica. In seguito, verrà creata una ProductsOptimisticConcurrencyBLL classe Livello di logica di business che include le modifiche appropriate per supportare la concorrenza ottimistica DAL. Dopo aver posato questo lavoro, saremo pronti per creare la pagina ASP.NET.

Passaggio 2: Creazione di un livello di accesso ai dati che supporta la concorrenza ottimistica

Per creare un nuovo DataSet tipizzato, fare clic con il pulsante destro del mouse sulla cartella all'interno della DALApp_Code cartella e aggiungere un nuovo Oggetto DataSet denominato NorthwindOptimisticConcurrency. Come illustrato nella prima esercitazione, in questo modo si aggiungerà un nuovo TableAdapter al DataSet tipizzato, avviando automaticamente la Configurazione guidata TableAdapter. Nella prima schermata viene richiesto di specificare il database a cui connettersi, connettersi allo stesso database Northwind usando l'impostazione NORTHWNDConnectionString da Web.config.

Connettersi allo stesso database Northwind

Figura 3: Connettersi allo stesso database Northwind (Fare clic per visualizzare l'immagine full-size)

Verrà quindi richiesto come eseguire query sui dati: tramite un'istruzione SQL ad hoc, una nuova stored procedure o una stored procedure esistente. Poiché sono state usate query SQL ad hoc nel dal dal originale, usare questa opzione anche qui.

Specificare i dati da recuperare usando un'istruzione SQL ad hoc

Figura 4: Specificare i dati da recuperare usando un'istruzione SQL ad hoc (fare clic per visualizzare l'immagine full-size)

Nella schermata seguente immettere la query SQL da usare per recuperare le informazioni sul prodotto. Si userà la stessa query SQL utilizzata per TableAdapter Products dall'originale DAL, che restituisce tutte le colonne insieme ai nomi dei fornitori e delle Product categorie del prodotto:

SELECT   ProductID, ProductName, SupplierID, CategoryID, QuantityPerUnit,
           UnitPrice, UnitsInStock, UnitsOnOrder, ReorderLevel, Discontinued,
           (SELECT CategoryName FROM Categories
              WHERE Categories.CategoryID = Products.CategoryID)
              as CategoryName,
           (SELECT CompanyName FROM Suppliers
              WHERE Suppliers.SupplierID = Products.SupplierID)
              as SupplierName
FROM     Products

Usare la stessa query SQL da Products TableAdapter nel DAL originale

Figura 5: Usare la stessa query SQL da Products TableAdapter nel DAL originale (fare clic per visualizzare l'immagine a dimensioni complete)

Prima di passare alla schermata successiva, fare clic sul pulsante Opzioni avanzate. Per avere questo controllo TableAdapter che usa il controllo di concorrenza ottimistica, selezionare semplicemente la casella di controllo "Usa concorrenza ottimistica".

Abilitare il controllo di concorrenza ottimistica controllando la casella di controllo

Figura 6: Abilitare il controllo di concorrenza ottimistica controllando la casella di controllo "Usa concorrenza ottimistica" (fare clic per visualizzare l'immagine full-size)

Infine, indicare che TableAdapter deve usare i modelli di accesso ai dati che riempiono una tabella dati e restituiscono una tabella dati; indicare anche che i metodi diretti del database devono essere creati. Modificare il nome del metodo per il modello Return a DataTable da GetData a GetProducts, in modo da eseguire il mirroring delle convenzioni di denominazione usate nel DAL originale.

Usare tutti i modelli di accesso ai dati di TableAdapter

Figura 7: Usare tutti i modelli di accesso ai dati (fare clic per visualizzare l'immagine full-size)

Al termine della procedura guidata, l'Designer DataSet includerà una tabella dati fortemente tipizzata Products e TableAdapter. Per rinominare DataTable da Products a ProductsOptimisticConcurrency, è possibile fare clic con il pulsante destro del mouse sulla barra del titolo di DataTable e scegliere Rinomina dal menu di scelta rapida.

Un oggetto DataTable e TableAdapter sono stati aggiunti al dataset tipizzato

Figura 8: Una tabella dati e TableAdapter sono stati aggiunti al dataset tipizzato (fare clic per visualizzare l'immagine full-size)

Per visualizzare le differenze tra le UPDATE query e DELETE tra TableAdapter (che usa la ProductsOptimisticConcurrency concorrenza ottimistica) e Products TableAdapter (che non è), fare clic su TableAdapter e passare alla Finestra Proprietà. DeleteCommand Nelle proprietà delle proprietà e UpdateCommandCommandText è possibile visualizzare la sintassi SQL effettiva inviata al database quando vengono richiamati i metodi correlati all'aggiornamento o all'eliminazione di DAL. Per l'istruzione ProductsOptimisticConcurrency TableAdapter DELETE usata è:

DELETE FROM [Products]
    WHERE (([ProductID] = @Original_ProductID)
    AND ([ProductName] = @Original_ProductName)
    AND ((@IsNull_SupplierID = 1 AND [SupplierID] IS NULL)
       OR ([SupplierID] = @Original_SupplierID))
    AND ((@IsNull_CategoryID = 1 AND [CategoryID] IS NULL)
       OR ([CategoryID] = @Original_CategoryID))
    AND ((@IsNull_QuantityPerUnit = 1 AND [QuantityPerUnit] IS NULL)
       OR ([QuantityPerUnit] = @Original_QuantityPerUnit))
    AND ((@IsNull_UnitPrice = 1 AND [UnitPrice] IS NULL)
       OR ([UnitPrice] = @Original_UnitPrice))
    AND ((@IsNull_UnitsInStock = 1 AND [UnitsInStock] IS NULL)
       OR ([UnitsInStock] = @Original_UnitsInStock))
    AND ((@IsNull_UnitsOnOrder = 1 AND [UnitsOnOrder] IS NULL)
       OR ([UnitsOnOrder] = @Original_UnitsOnOrder))
    AND ((@IsNull_ReorderLevel = 1 AND [ReorderLevel] IS NULL)
       OR ([ReorderLevel] = @Original_ReorderLevel))
    AND ([Discontinued] = @Original_Discontinued))

Mentre l'istruzione DELETE per Product TableAdapter nell'originale DAL è molto più semplice:

DELETE FROM [Products] WHERE (([ProductID] = @Original_ProductID))

Come si può notare, la clausola nell'istruzione DELETE per TableAdapter che usa la WHERE concorrenza ottimistica include un confronto tra i valori di Product colonna esistenti della tabella e i valori originali al momento in cui GridView (o DetailsView o FormView) è stato popolato. Poiché tutti i campi diversi da ProductID, ProductNamee Discontinued possono avere NULL valori, sono inclusi parametri e controlli aggiuntivi per confrontare NULL correttamente i valori nella WHERE clausola .

Per questa esercitazione non verranno aggiunti altri DataTables all'oggetto DataSet abilitato per la concorrenza ottimistica per questa esercitazione, poiché la pagina ASP.NET fornirà solo l'aggiornamento e l'eliminazione delle informazioni sul prodotto. Tuttavia, è comunque necessario aggiungere il metodo all'oggetto GetProductByProductID(productID)ProductsOptimisticConcurrency TableAdapter.

A questo scopo, fare clic con il pulsante destro del mouse sulla barra del titolo di TableAdapter (l'area sopra i Fill nomi dei metodi e GetProducts ) e scegliere Aggiungi query dal menu di scelta rapida. Verrà avviata la Configurazione guidata query TableAdapter. Come per la configurazione iniziale di TableAdapter, scegliere di creare il GetProductByProductID(productID) metodo usando un'istruzione SQL ad hoc (vedere la figura 4). Poiché il GetProductByProductID(productID) metodo restituisce informazioni su un determinato prodotto, indicare che questa query è un SELECT tipo di query che restituisce righe.

Contrassegnare il tipo di query come

Figura 9: Contrassegnare il tipo di query come "SELECT che restituisce righe" (fare clic per visualizzare l'immagine a dimensione intera)

Nella schermata successiva viene chiesto di usare la query SQL con la query predefinita di TableAdapter precaricata. Aumentare la query esistente per includere la clausola WHERE ProductID = @ProductID, come illustrato nella figura 10.

Aggiungere una clausola WHERE alla query precaricata per restituire un record prodotto specifico

Figura 10: Aggiungere una WHERE clausola alla query precaricata per restituire un record prodotto specifico (fare clic per visualizzare l'immagine a dimensione intera)

Modificare infine i nomi dei metodi generati in FillByProductID e GetProductByProductID.

Rinominare i metodi in FillByProductID e GetProductByProductID

Figura 11: Rinominare i metodi in FillByProductID e GetProductByProductID (fare clic per visualizzare l'immagine a dimensione intera)

Al termine della procedura guidata, TableAdapter contiene ora due metodi per il recupero dei dati: GetProducts(), che restituisce tutti i prodotti e GetProductByProductID(productID), che restituisce il prodotto specificato.

Passaggio 3: Creazione di un livello di logica di business per l'Concurrency-Enabled ottimistico dal

La classe esistente ProductsBLL include esempi di uso sia dell'aggiornamento batch che dei modelli diretti del database. Il AddProduct metodo e UpdateProduct gli overload usano entrambi il modello di aggiornamento batch, passando un'istanza ProductRow al metodo Update di TableAdapter. Il DeleteProduct metodo, d'altra parte, usa il modello diretto del database, chiamando il metodo TableAdapter Delete(productID) .

Con il nuovo ProductsOptimisticConcurrency TableAdapter, i metodi diretti del database richiedono ora che vengano passati anche i valori originali. Ad esempio, il Delete metodo prevede ora dieci parametri di input: originaleProductID, , ProductNameSupplierID, CategoryID, UnitPriceUnitsInStockUnitsOnOrderQuantityPerUnitReorderLevele .Discontinued Usa questi valori aggiuntivi dei parametri di input nella WHERE clausola dell'istruzione DELETE inviata al database, eliminando solo il record specificato se i valori correnti del database vengono mappati a quelli originali.

Anche se la firma del metodo per Update il metodo TableAdapter usato nel modello di aggiornamento batch non è stata modificata, il codice necessario per registrare i valori originali e nuovi ha. Di conseguenza, anziché tentare di usare il servizio di integrazione con concorrenza ottimistica con la classe esistente ProductsBLL , verrà creata una nuova classe Livello logica di business per l'uso del nuovo DAL.

Aggiungere una classe denominata ProductsOptimisticConcurrencyBLL alla BLL cartella all'interno della App_Code cartella .

Aggiungere la classe ProductsOptimisticConcurrencyBLL alla cartella BLL

Figura 12: Aggiungere la ProductsOptimisticConcurrencyBLL classe alla cartella BLL

Aggiungere quindi il codice seguente alla ProductsOptimisticConcurrencyBLL classe :

using System;
using System.Data;
using System.Configuration;
using System.Web;
using System.Web.Security;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Web.UI.WebControls.WebParts;
using System.Web.UI.HtmlControls;
using NorthwindOptimisticConcurrencyTableAdapters;
[System.ComponentModel.DataObject]
public class ProductsOptimisticConcurrencyBLL
{
    private ProductsOptimisticConcurrencyTableAdapter _productsAdapter = null;
    protected ProductsOptimisticConcurrencyTableAdapter Adapter
    {
        get
        {
            if (_productsAdapter == null)
                _productsAdapter = new ProductsOptimisticConcurrencyTableAdapter();
            return _productsAdapter;
        }
    }
    [System.ComponentModel.DataObjectMethodAttribute
    (System.ComponentModel.DataObjectMethodType.Select, true)]
    public NorthwindOptimisticConcurrency.ProductsOptimisticConcurrencyDataTable GetProducts()
    {
        return Adapter.GetProducts();
    }
}

Si noti l'istruzione using NorthwindOptimisticConcurrencyTableAdapters sopra l'inizio della dichiarazione di classe. Lo NorthwindOptimisticConcurrencyTableAdapters spazio dei nomi contiene la ProductsOptimisticConcurrencyTableAdapter classe , che fornisce i metodi di DAL. Prima della dichiarazione di classe si troverà anche l'attributo System.ComponentModel.DataObject , che indica a Visual Studio di includere questa classe nell'elenco a discesa della procedura guidata ObjectDataSource.

La ProductsOptimisticConcurrencyBLLproprietà della Adapter classe fornisce accesso rapido a un'istanza della ProductsOptimisticConcurrencyTableAdapter classe e segue il modello usato nelle classi BLL originali (ProductsBLL, CategoriesBLLe così via). Infine, il GetProducts() metodo chiama semplicemente nel metodo DAL GetProducts() e restituisce un ProductsOptimisticConcurrencyDataTable oggetto popolato con un'istanza ProductsOptimisticConcurrencyRow per ogni record di prodotto nel database.

Eliminazione di un prodotto usando il modello diretto del database con concorrenza ottimistica

Quando si usa il modello diretto del database rispetto a un'istanza dal che usa la concorrenza ottimistica, i metodi devono essere passati ai valori nuovi e originali. Per l'eliminazione, non sono presenti nuovi valori, quindi è necessario passare solo i valori originali. In BLL, quindi, è necessario accettare tutti i parametri originali come parametri di input. Si avrà il DeleteProduct metodo nella ProductsOptimisticConcurrencyBLL classe che usa il metodo diretto del database. Ciò significa che questo metodo deve accettare tutti e dieci i campi dati del prodotto come parametri di input e passarli a DAL, come illustrato nel codice seguente:

[System.ComponentModel.DataObjectMethodAttribute
(System.ComponentModel.DataObjectMethodType.Delete, true)]
public bool DeleteProduct
    (int original_productID, string original_productName,
    int? original_supplierID, int? original_categoryID,
    string original_quantityPerUnit, decimal? original_unitPrice,
    short? original_unitsInStock, short? original_unitsOnOrder,
    short? original_reorderLevel, bool original_discontinued)
{
    int rowsAffected = Adapter.Delete(original_productID,
                                      original_productName,
                                      original_supplierID,
                                      original_categoryID,
                                      original_quantityPerUnit,
                                      original_unitPrice,
                                      original_unitsInStock,
                                      original_unitsOnOrder,
                                      original_reorderLevel,
                                      original_discontinued);
    // Return true if precisely one row was deleted, otherwise false
    return rowsAffected == 1;
}

Se i valori originali, ovvero quelli che sono stati caricati per l'ultima volta in GridView (o DetailsView o FormView), differiscono dai valori nel database quando l'utente fa clic sul pulsante Elimina la WHERE clausola non corrisponderà ad alcun record di database e non saranno interessati record. Di conseguenza, il metodo di Delete TableAdapter restituirà 0 e il metodo BLL DeleteProduct restituirà false.

Aggiornamento di un prodotto tramite il modello di aggiornamento batch con concorrenza ottimistica

Come indicato in precedenza, il metodo tableAdapter Update per il modello di aggiornamento batch ha la stessa firma del metodo indipendentemente dal fatto che venga usata o meno la concorrenza ottimistica. In genere, il Update metodo prevede un DataRow, una matrice di DataRows, un Oggetto DataTable o un DataSet tipizzato. Non sono disponibili parametri di input aggiuntivi per specificare i valori originali. Ciò è possibile perché DataTable tiene traccia dei valori originali e modificati per i relativi datarow.this is possible because the DataTable tiene traccia dei valori originali e modificati per i relativi DataRow/s. Quando il dal rilascia UPDATE l'istruzione , i @original_ColumnName parametri vengono popolati con i valori originali di DataRow, mentre i @ColumnName parametri vengono popolati con i valori modificati di DataRow.

ProductsBLL Nella classe (che usa il modello di concorrenza originale e non ottimistica), quando si usa il modello di aggiornamento batch per aggiornare le informazioni sul prodotto, il codice esegue la sequenza di eventi seguente:

  1. Leggere le informazioni sul prodotto del database corrente in un'istanza ProductRow usando il metodo TableAdapter GetProductByProductID(productID)
  2. Assegnare i nuovi valori all'istanza ProductRow del passaggio 1
  3. Chiamare il metodo tableAdapterUpdate, passando l'istanza ProductRow

Questa sequenza di passaggi, tuttavia, non supporterà correttamente la concorrenza ottimistica perché il ProductRow popolamento nel passaggio 1 viene popolato direttamente dal database, ovvero i valori originali usati da DataRow sono quelli attualmente presenti nel database e non quelli associati a GridView all'inizio del processo di modifica. Quando invece si usa un'istanza DAL abilitata per la concorrenza ottimistica, è necessario modificare gli overload del UpdateProduct metodo per usare la procedura seguente:

  1. Leggere le informazioni sul prodotto del database corrente in un'istanza ProductsOptimisticConcurrencyRow usando il metodo TableAdapter GetProductByProductID(productID)
  2. Assegnare i valori originali all'istanza ProductsOptimisticConcurrencyRow del passaggio 1
  3. Chiamare il ProductsOptimisticConcurrencyRow metodo dell'istanza AcceptChanges() , che indica a DataRow che i valori correnti sono quelli "originali"
  4. Assegnare i nuovi valori all'istanza ProductsOptimisticConcurrencyRow
  5. Chiamare il metodo tableAdapterUpdate, passando l'istanza ProductsOptimisticConcurrencyRow

Il passaggio 1 legge tutti i valori di database correnti per il record prodotto specificato. Questo passaggio è superfluo nell'overload UpdateProduct che aggiorna tutte le colonne del prodotto (poiché questi valori vengono sovrascritti nel passaggio 2), ma è essenziale per gli overload in cui solo un subset dei valori di colonna vengono passati come parametri di input. Dopo aver assegnato i valori originali all'istanza ProductsOptimisticConcurrencyRow , viene chiamato il AcceptChanges() metodo , che contrassegna i valori DataRow correnti come valori originali da usare nei @original_ColumnName parametri dell'istruzione UPDATE . Successivamente, i nuovi valori dei parametri vengono assegnati a ProductsOptimisticConcurrencyRow e, infine, il Update metodo viene richiamato, passando il DataRow.

Il codice seguente mostra l'overload UpdateProduct che accetta tutti i campi dati del prodotto come parametri di input. Anche se non illustrato qui, la ProductsOptimisticConcurrencyBLL classe inclusa nel download per questa esercitazione contiene anche un UpdateProduct overload che accetta solo il nome e il prezzo del prodotto come parametri di input.

protected void AssignAllProductValues
    (NorthwindOptimisticConcurrency.ProductsOptimisticConcurrencyRow product,
    string productName, int? supplierID, int? categoryID, string quantityPerUnit,
    decimal? unitPrice, short? unitsInStock, short? unitsOnOrder,
    short? reorderLevel, bool discontinued)
{
    product.ProductName = productName;
    if (supplierID == null)
        product.SetSupplierIDNull();
    else
        product.SupplierID = supplierID.Value;
    if (categoryID == null)
        product.SetCategoryIDNull();
    else
        product.CategoryID = categoryID.Value;
    if (quantityPerUnit == null)
        product.SetQuantityPerUnitNull();
    else
        product.QuantityPerUnit = quantityPerUnit;
    if (unitPrice == null)
        product.SetUnitPriceNull();
    else
        product.UnitPrice = unitPrice.Value;
    if (unitsInStock == null)
        product.SetUnitsInStockNull();
    else
        product.UnitsInStock = unitsInStock.Value;
    if (unitsOnOrder == null)
        product.SetUnitsOnOrderNull();
    else
        product.UnitsOnOrder = unitsOnOrder.Value;
    if (reorderLevel == null)
        product.SetReorderLevelNull();
    else
        product.ReorderLevel = reorderLevel.Value;
    product.Discontinued = discontinued;
}
[System.ComponentModel.DataObjectMethodAttribute
(System.ComponentModel.DataObjectMethodType.Update, true)]
public bool UpdateProduct(
    // new parameter values
    string productName, int? supplierID, int? categoryID, string quantityPerUnit,
    decimal? unitPrice, short? unitsInStock, short? unitsOnOrder,
    short? reorderLevel, bool discontinued, int productID,
    // original parameter values
    string original_productName, int? original_supplierID, int? original_categoryID,
    string original_quantityPerUnit, decimal? original_unitPrice,
    short? original_unitsInStock, short? original_unitsOnOrder,
    short? original_reorderLevel, bool original_discontinued,
    int original_productID)
{
    // STEP 1: Read in the current database product information
    NorthwindOptimisticConcurrency.ProductsOptimisticConcurrencyDataTable products =
        Adapter.GetProductByProductID(original_productID);
    if (products.Count == 0)
        // no matching record found, return false
        return false;
    NorthwindOptimisticConcurrency.ProductsOptimisticConcurrencyRow product = products[0];
    // STEP 2: Assign the original values to the product instance
    AssignAllProductValues(product, original_productName, original_supplierID,
        original_categoryID, original_quantityPerUnit, original_unitPrice,
        original_unitsInStock, original_unitsOnOrder, original_reorderLevel,
        original_discontinued);
    // STEP 3: Accept the changes
    product.AcceptChanges();
    // STEP 4: Assign the new values to the product instance
    AssignAllProductValues(product, productName, supplierID, categoryID,
        quantityPerUnit, unitPrice, unitsInStock, unitsOnOrder, reorderLevel,
        discontinued);
    // STEP 5: Update the product record
    int rowsAffected = Adapter.Update(product);
    // Return true if precisely one row was updated, otherwise false
    return rowsAffected == 1;
}

Passaggio 4: Passaggio dei valori originali e nuovi dalla pagina ASP.NET ai metodi BLL

Con il completamento di DAL e BLL, tutto ciò che rimane consiste nel creare una pagina di ASP.NET in grado di usare la logica di concorrenza ottimistica integrata nel sistema. In particolare, il controllo Web dei dati (GridView, DetailsView o FormView) deve ricordare i valori originali e ObjectDataSource deve passare entrambi i set di valori al livello della logica di business. Inoltre, la pagina ASP.NET deve essere configurata per gestire correttamente le violazioni della concorrenza.

Per iniziare, aprire la OptimisticConcurrency.aspx pagina nella EditInsertDelete cartella e aggiungere un controllo GridView al Designer, impostandone la ID proprietà su ProductsGrid. Dallo smart tag di GridView scegliere di creare un nuovo ObjectDataSource denominato ProductsOptimisticConcurrencyDataSource. Poiché si vuole che ObjectDataSource usi il dal che supporta la concorrenza ottimistica, configurarlo per l'uso dell'oggetto ProductsOptimisticConcurrencyBLL .

Fare in modo che ObjectDataSource usi l'oggetto ProductsOptimisticConcurrencyBLL

Figura 13: Usare ObjectDataSource (Fare clic per visualizzare l'immagineProductsOptimisticConcurrencyBLL a dimensione intera)

Scegliere i GetProductsmetodi , UpdateProducte DeleteProduct dagli elenchi a discesa della procedura guidata. Per il metodo UpdateProduct, utilizzare l'overload che accetta tutti i campi dati del prodotto.

Configurazione delle proprietà del controllo ObjectDataSource

Al termine della procedura guidata, il markup dichiarativo di ObjectDataSource avrà un aspetto simile al seguente:

<asp:ObjectDataSource ID="ProductsOptimisticConcurrencyDataSource" runat="server"
    DeleteMethod="DeleteProduct" OldValuesParameterFormatString="original_{0}"
    SelectMethod="GetProducts" TypeName="ProductsOptimisticConcurrencyBLL"
    UpdateMethod="UpdateProduct">
    <DeleteParameters>
        <asp:Parameter Name="original_productID" Type="Int32" />
        <asp:Parameter Name="original_productName" Type="String" />
        <asp:Parameter Name="original_supplierID" Type="Int32" />
        <asp:Parameter Name="original_categoryID" Type="Int32" />
        <asp:Parameter Name="original_quantityPerUnit" Type="String" />
        <asp:Parameter Name="original_unitPrice" Type="Decimal" />
        <asp:Parameter Name="original_unitsInStock" Type="Int16" />
        <asp:Parameter Name="original_unitsOnOrder" Type="Int16" />
        <asp:Parameter Name="original_reorderLevel" Type="Int16" />
        <asp:Parameter Name="original_discontinued" Type="Boolean" />
    </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" />
        <asp:Parameter Name="original_productName" Type="String" />
        <asp:Parameter Name="original_supplierID" Type="Int32" />
        <asp:Parameter Name="original_categoryID" Type="Int32" />
        <asp:Parameter Name="original_quantityPerUnit" Type="String" />
        <asp:Parameter Name="original_unitPrice" Type="Decimal" />
        <asp:Parameter Name="original_unitsInStock" Type="Int16" />
        <asp:Parameter Name="original_unitsOnOrder" Type="Int16" />
        <asp:Parameter Name="original_reorderLevel" Type="Int16" />
        <asp:Parameter Name="original_discontinued" Type="Boolean" />
        <asp:Parameter Name="original_productID" Type="Int32" />
    </UpdateParameters>
</asp:ObjectDataSource>

Come si può notare, la raccolta contiene un'istanza DeleteParametersParameter per ognuno dei dieci parametri di input nel ProductsOptimisticConcurrencyBLL metodo della DeleteProduct classe. Analogamente, la raccolta contiene un'istanza UpdateParametersParameter per ognuno dei parametri di input in UpdateProduct.

Per le esercitazioni precedenti che hanno comportato la modifica dei dati, rimuoveremo la proprietà ObjectDataSource OldValuesParameterFormatString a questo punto, poiché questa proprietà indica che il metodo BLL prevede che i valori precedenti (o originali) vengano passati, nonché i nuovi valori. Inoltre, questo valore della proprietà indica i nomi dei parametri di input per i valori originali. Poiché i valori originali vengono passati al BLL, non rimuovere questa proprietà.

Nota

Il valore della OldValuesParameterFormatString proprietà deve essere mappato ai nomi dei parametri di input nel BLL che prevedono i valori originali. Poiché questi parametri sono stati denominati original_productName, original_supplierIDe così via, è possibile lasciare il valore della OldValuesParameterFormatString proprietà come original_{0}. Se, tuttavia, i parametri di input dei metodi BLL hanno nomi come old_productName, old_supplierIDe così via, è necessario aggiornare la OldValuesParameterFormatString proprietà a old_{0}.

È necessario eseguire un'impostazione finale della proprietà affinché ObjectDataSource passi correttamente i valori originali ai metodi BLL. ObjectDataSource ha una proprietà ConflictDetection che può essere assegnata a uno dei due valori seguenti:

  • OverwriteChanges - il valore predefinito; non invia i valori originali ai parametri di input originali dei metodi BLL
  • CompareAllValues - invia i valori originali ai metodi BLL; scegliere questa opzione quando si usa la concorrenza ottimistica

Impostare la ConflictDetection proprietà su CompareAllValues.

Configurazione delle proprietà e dei campi di GridView

Con le proprietà di ObjectDataSource configurate correttamente, è possibile assegnare l'attenzione alla configurazione di GridView. In primo luogo, poiché gridView supporta la modifica e l'eliminazione, fare clic sulle caselle di controllo Abilita modifica e Abilita eliminazione dallo smart tag di GridView. Verrà aggiunto un campo CommandField il cui ShowEditButton e sono ShowDeleteButton entrambi impostati su true.

Se associato a ProductsOptimisticConcurrencyDataSource ObjectDataSource, GridView contiene un campo per ognuno dei campi dati del prodotto. Anche se è possibile modificare un controllo GridView di questo tipo, l'esperienza utente è accettabile. Il rendering di CategoryID e SupplierID BoundFields verrà eseguito come TextBoxes, richiedendo all'utente di immettere la categoria e il fornitore appropriati come numeri ID. Non ci sarà alcuna formattazione per i campi numerici e nessun controllo di convalida per garantire che il nome del prodotto sia stato fornito e che il prezzo unitario, le unità in magazzino, le unità in ordine e i valori del livello di riordinamento siano sia valori numerici appropriati che siano maggiori o uguali a zero.

Come illustrato nell'esercitazione Aggiunta di controlli di convalida alle interfacce di modifica e inserimento e personalizzazione delle interfacce di modifica dei dati , l'interfaccia utente può essere personalizzata sostituendo BoundFields con TemplateFields. L'interfaccia di modifica gridView è stata modificata nei modi seguenti:

  • Rimozione di ProductID, SupplierNamee CategoryName BoundFields
  • Converte boundField ProductName in un campo template e aggiunge un controllo RequiredFieldValidation.
  • Convertire e CategoryIDSupplierID BoundFields in TemplateFields e modificare l'interfaccia di modifica in modo da usare DropDownLists anziché TextBoxes. In questi campi TemplateFields vengono ItemTemplatesvisualizzati i CategoryName campi dati e SupplierName .
  • Convertito i UnitPricecontrolli , UnitsInStock, UnitsOnOrdere ReorderLevel BoundFields in TemplateFields e aggiunti i controlli CompareValidator.

Poiché è già stato esaminato come eseguire queste attività nelle esercitazioni precedenti, verrà elencata solo la sintassi dichiarativa finale qui e lasciare l'implementazione come pratica.

<asp:GridView ID="ProductsGrid" runat="server" AutoGenerateColumns="False"
    DataKeyNames="ProductID" DataSourceID="ProductsOptimisticConcurrencyDataSource"
    OnRowUpdated="ProductsGrid_RowUpdated">
    <Columns>
        <asp:CommandField ShowDeleteButton="True" ShowEditButton="True" />
        <asp:TemplateField HeaderText="Product" SortExpression="ProductName">
            <EditItemTemplate>
                <asp:TextBox ID="EditProductName" runat="server"
                    Text='<%# Bind("ProductName") %>'></asp:TextBox>
                <asp:RequiredFieldValidator ID="RequiredFieldValidator1"
                    ControlToValidate="EditProductName"
                    ErrorMessage="You must enter a product name."
                    runat="server">*</asp:RequiredFieldValidator>
            </EditItemTemplate>
            <ItemTemplate>
                <asp:Label ID="Label1" runat="server"
                    Text='<%# Bind("ProductName") %>'></asp:Label>
            </ItemTemplate>
        </asp:TemplateField>
        <asp:TemplateField HeaderText="Category" SortExpression="CategoryName">
            <EditItemTemplate>
                <asp:DropDownList ID="EditCategoryID" runat="server"
                    DataSourceID="CategoriesDataSource" AppendDataBoundItems="true"
                    DataTextField="CategoryName" DataValueField="CategoryID"
                    SelectedValue='<%# Bind("CategoryID") %>'>
                    <asp:ListItem Value=">(None)</asp:ListItem>
                </asp:DropDownList><asp:ObjectDataSource ID="CategoriesDataSource"
                    runat="server" OldValuesParameterFormatString="original_{0}"
                    SelectMethod="GetCategories" TypeName="CategoriesBLL">
                </asp:ObjectDataSource>
            </EditItemTemplate>
            <ItemTemplate>
                <asp:Label ID="Label2" runat="server"
                    Text='<%# Bind("CategoryName") %>'></asp:Label>
            </ItemTemplate>
        </asp:TemplateField>
        <asp:TemplateField HeaderText="Supplier" SortExpression="SupplierName">
            <EditItemTemplate>
                <asp:DropDownList ID="EditSuppliersID" runat="server"
                    DataSourceID="SuppliersDataSource" AppendDataBoundItems="true"
                    DataTextField="CompanyName" DataValueField="SupplierID"
                    SelectedValue='<%# Bind("SupplierID") %>'>
                    <asp:ListItem Value=">(None)</asp:ListItem>
                </asp:DropDownList><asp:ObjectDataSource ID="SuppliersDataSource"
                    runat="server" OldValuesParameterFormatString="original_{0}"
                    SelectMethod="GetSuppliers" TypeName="SuppliersBLL">
                </asp:ObjectDataSource>
            </EditItemTemplate>
            <ItemTemplate>
                <asp:Label ID="Label3" runat="server"
                    Text='<%# Bind("SupplierName") %>'></asp:Label>
            </ItemTemplate>
        </asp:TemplateField>
        <asp:BoundField DataField="QuantityPerUnit" HeaderText="Qty/Unit"
            SortExpression="QuantityPerUnit" />
        <asp:TemplateField HeaderText="Price" SortExpression="UnitPrice">
            <EditItemTemplate>
                <asp:TextBox ID="EditUnitPrice" runat="server"
                    Text='<%# Bind("UnitPrice", "{0:N2}") %>' Columns="8" />
                <asp:CompareValidator ID="CompareValidator1" runat="server"
                    ControlToValidate="EditUnitPrice"
                    ErrorMessage="Unit price must be a valid currency value without the
                    currency symbol and must have a value greater than or equal to zero."
                    Operator="GreaterThanEqual" Type="Currency"
                    ValueToCompare="0">*</asp:CompareValidator>
            </EditItemTemplate>
            <ItemTemplate>
                <asp:Label ID="Label4" runat="server"
                    Text='<%# Bind("UnitPrice", "{0:C}") %>'></asp:Label>
            </ItemTemplate>
        </asp:TemplateField>
        <asp:TemplateField HeaderText="Units In Stock" SortExpression="UnitsInStock">
            <EditItemTemplate>
                <asp:TextBox ID="EditUnitsInStock" runat="server"
                    Text='<%# Bind("UnitsInStock") %>' Columns="6"></asp:TextBox>
                <asp:CompareValidator ID="CompareValidator2" runat="server"
                    ControlToValidate="EditUnitsInStock"
                    ErrorMessage="Units in stock must be a valid number
                        greater than or equal to zero."
                    Operator="GreaterThanEqual" Type="Integer"
                    ValueToCompare="0">*</asp:CompareValidator>
            </EditItemTemplate>
            <ItemTemplate>
                <asp:Label ID="Label5" runat="server"
                    Text='<%# Bind("UnitsInStock", "{0:N0}") %>'></asp:Label>
            </ItemTemplate>
        </asp:TemplateField>
        <asp:TemplateField HeaderText="Units On Order" SortExpression="UnitsOnOrder">
            <EditItemTemplate>
                <asp:TextBox ID="EditUnitsOnOrder" runat="server"
                    Text='<%# Bind("UnitsOnOrder") %>' Columns="6"></asp:TextBox>
                <asp:CompareValidator ID="CompareValidator3" runat="server"
                    ControlToValidate="EditUnitsOnOrder"
                    ErrorMessage="Units on order must be a valid numeric value
                        greater than or equal to zero."
                    Operator="GreaterThanEqual" Type="Integer"
                    ValueToCompare="0">*</asp:CompareValidator>
            </EditItemTemplate>
            <ItemTemplate>
                <asp:Label ID="Label6" runat="server"
                    Text='<%# Bind("UnitsOnOrder", "{0:N0}") %>'></asp:Label>
            </ItemTemplate>
        </asp:TemplateField>
        <asp:TemplateField HeaderText="Reorder Level" SortExpression="ReorderLevel">
            <EditItemTemplate>
                <asp:TextBox ID="EditReorderLevel" runat="server"
                    Text='<%# Bind("ReorderLevel") %>' Columns="6"></asp:TextBox>
                <asp:CompareValidator ID="CompareValidator4" runat="server"
                    ControlToValidate="EditReorderLevel"
                    ErrorMessage="Reorder level must be a valid numeric value
                        greater than or equal to zero."
                    Operator="GreaterThanEqual" Type="Integer"
                    ValueToCompare="0">*</asp:CompareValidator>
            </EditItemTemplate>
            <ItemTemplate>
                <asp:Label ID="Label7" runat="server"
                    Text='<%# Bind("ReorderLevel", "{0:N0}") %>'></asp:Label>
            </ItemTemplate>
        </asp:TemplateField>
        <asp:CheckBoxField DataField="Discontinued" HeaderText="Discontinued"
            SortExpression="Discontinued" />
    </Columns>
</asp:GridView>

Siamo molto vicini ad avere un esempio completamente funzionante. Tuttavia, ci sono alcune sottigliezze che si inquieteranno e ci causeranno problemi. È inoltre necessaria un'interfaccia che avvisa l'utente quando si è verificata una violazione della concorrenza.

Nota

Affinché un controllo Web dati passi correttamente i valori originali a ObjectDataSource (che vengono quindi passati al BLL), è fondamentale che la proprietà di EnableViewState GridView sia impostata su true (impostazione predefinita). Se si disabilita lo stato di visualizzazione, i valori originali vengono persi al postback.

Passaggio dei valori originali corretti a ObjectDataSource

Ci sono un paio di problemi con il modo in cui GridView è stato configurato. Se la proprietà ObjectDataSource ConflictDetection è impostata su CompareAllValues (come avviene con il nostro), quando i metodi o Delete() di ObjectDataSource vengono richiamati da GridView (o DetailsView o FormView), ObjectDataSource tenta di copiare i valori originali di Update() GridView nelle istanze appropriateParameter. Fare riferimento alla figura 2 per una rappresentazione grafica di questo processo.

In particolare, ai valori originali di GridView vengono assegnati i valori nelle istruzioni databinding bidirezionali ogni volta che i dati sono associati a GridView. Di conseguenza, è essenziale che tutti i valori originali richiesti vengano acquisiti tramite databinding bidirezionale e che vengano forniti in un formato convertibile.

Per vedere perché questo è importante, prendere un momento per visitare la nostra pagina in un browser. Come previsto, GridView elenca ogni prodotto con un pulsante Modifica ed Elimina nella colonna più a sinistra.

I prodotti sono elencati in un controllo GridView

Figura 14: I prodotti sono elencati in un controllo GridView (fare clic per visualizzare l'immagine a dimensione intera)

Se si fa clic sul pulsante Elimina per qualsiasi prodotto, viene generata un'eccezione FormatException .

Tentativo di eliminare i risultati di un prodotto in un'eccezione FormatException

Figura 15: Tentativo di eliminare i risultati di un prodotto in un oggetto FormatException (fare clic per visualizzare un'immagine di dimensioni intere)

Viene FormatException generato quando ObjectDataSource tenta di leggere nel valore originale UnitPrice . Poiché l'oggetto ItemTemplateUnitPrice è formattato come valuta (<%# Bind("UnitPrice", "{0:C}") %>), include un simbolo di valuta, ad esempio $19,95. Si FormatException verifica quando ObjectDataSource tenta di convertire questa stringa in un oggetto decimal. Per aggirare questo problema, sono disponibili diverse opzioni:

  • Rimuovere la formattazione della valuta da ItemTemplate. Invece di usare <%# Bind("UnitPrice", "{0:C}") %>, è sufficiente usare <%# Bind("UnitPrice") %>. Lo svantaggio di questo è che il prezzo non è più formattato.
  • Visualizzare la UnitPrice classe formattata come valuta in ItemTemplate, ma usare la Eval parola chiave per eseguire questa operazione. Tenere presente che Eval esegue l'associazione dati unidirezionale. È comunque necessario specificare il UnitPrice valore per i valori originali, quindi sarà comunque necessaria un'istruzione databinding bidirezionale in ItemTemplate, ma può essere inserita in un controllo Web Label la cui Visible proprietà è impostata su false. È possibile usare il markup seguente in ItemTemplate:
<ItemTemplate>
    <asp:Label ID="DummyUnitPrice" runat="server"
        Text='<%# Bind("UnitPrice") %>' Visible="false"></asp:Label>
    <asp:Label ID="Label4" runat="server"
        Text='<%# Eval("UnitPrice", "{0:C}") %>'></asp:Label>
</ItemTemplate>
  • Rimuovere la formattazione della valuta da ItemTemplate, usando <%# Bind("UnitPrice") %>. Nel gestore eventi di GridView accedere a livello di RowDataBound codice al controllo Web Etichetta all'interno del quale viene visualizzato il valore e impostarne Text la UnitPrice proprietà sulla versione formattata.
  • Lasciare formattato UnitPrice come valuta. Nel gestore eventi di RowDeleting GridView sostituire il valore originale UnitPrice esistente ($19,95) con un valore decimale effettivo usando Decimal.Parse. È stato illustrato come eseguire un'operazione simile nel RowUpdating gestore eventi nell'esercitazione Sulla gestione di BLL- e DAL-Level eccezioni in un'esercitazione ASP.NET Page .

Per l'esempio scelto di procedere con il secondo approccio, aggiungendo un controllo Web Etichetta nascosta la cui Text proprietà è dati bidirezionali associati al valore non formattato UnitPrice .

Dopo aver risolto questo problema, provare a fare di nuovo clic sul pulsante Elimina per qualsiasi prodotto. Questa volta si otterrà un oggetto InvalidOperationException quando ObjectDataSource tenta di richiamare il metodo BLL UpdateProduct .

ObjectDataSource non è in grado di trovare un metodo con i parametri di input da inviare

Figura 16: ObjectDataSource non è in grado di trovare un metodo con i parametri di input da inviare (fare clic per visualizzare l'immagine a dimensione intera)

Esaminando il messaggio dell'eccezione, è chiaro che ObjectDataSource vuole richiamare un metodo BLL DeleteProduct che include original_CategoryName i parametri di input e original_SupplierName . Ciò è dovuto al fatto che gli ItemTemplate oggetti per CategoryID e SupplierID TemplateFields contengono attualmente istruzioni Bind bidirezionali con i CategoryName campi dati e SupplierName . È invece necessario includere Bind istruzioni con i CategoryID campi dati e SupplierID . A tale scopo, sostituire le istruzioni Bind esistenti con Eval istruzioni e quindi aggiungere controlli Label nascosti le CategoryID cui Text proprietà sono associate ai campi dati e SupplierID usando il databinding bidirezionale, come illustrato di seguito:

<asp:TemplateField HeaderText="Category" SortExpression="CategoryName">
    <EditItemTemplate>
        ...
    </EditItemTemplate>
    <ItemTemplate>
        <asp:Label ID="DummyCategoryID" runat="server"
            Text='<%# Bind("CategoryID") %>' Visible="False"></asp:Label>
        <asp:Label ID="Label2" runat="server"
            Text='<%# Eval("CategoryName") %>'></asp:Label>
    </ItemTemplate>
</asp:TemplateField>
<asp:TemplateField HeaderText="Supplier" SortExpression="SupplierName">
    <EditItemTemplate>
        ...
    </EditItemTemplate>
    <ItemTemplate>
        <asp:Label ID="DummySupplierID" runat="server"
            Text='<%# Bind("SupplierID") %>' Visible="False"></asp:Label>
        <asp:Label ID="Label3" runat="server"
            Text='<%# Eval("SupplierName") %>'></asp:Label>
    </ItemTemplate>
</asp:TemplateField>

Con queste modifiche, ora siamo in grado di eliminare e modificare correttamente le informazioni sul prodotto. Nel passaggio 5 si esaminerà come verificare che vengano rilevate violazioni della concorrenza. Per il momento, tuttavia, è necessario attendere alcuni minuti per provare ad aggiornare ed eliminare alcuni record per assicurarsi che l'aggiornamento e l'eliminazione per un singolo utente funzioni come previsto.

Passaggio 5: Test del supporto della concorrenza ottimistica

Per verificare che vengano rilevate violazioni della concorrenza (invece di comportare la sovrascrittura dei dati), è necessario aprire due finestre del browser in questa pagina. In entrambe le istanze del browser fare clic sul pulsante Modifica per Chai. Quindi, in uno solo dei browser, modificare il nome in "Chai Tea" e fare clic su Aggiorna. L'aggiornamento dovrebbe avere esito positivo e restituire GridView allo stato di pre-modifica, con "Chai Tea" come nuovo nome del prodotto.

Nell'altra istanza della finestra del browser, tuttavia, il nome del prodotto TextBox visualizza ancora "Chai". In questa seconda finestra del browser aggiornare in UnitPrice25.00. Senza il supporto della concorrenza ottimistica, facendo clic sull'aggiornamento nella seconda istanza del browser il nome del prodotto verrebbe modificato in "Chai", sovrascrivendo così le modifiche apportate dalla prima istanza del browser. Con la concorrenza ottimistica usata, tuttavia, facendo clic sul pulsante Aggiorna nella seconda istanza del browser viene generata un'eccezione DBConcurrencyException.

Quando viene rilevata una violazione della concorrenza, viene generata un'eccezione DBConcurrencyException

Figura 17: Quando viene rilevata una violazione della concorrenza, viene generata un'eccezione DBConcurrencyException (fare clic per visualizzare un'immagine a dimensione intera)

Viene DBConcurrencyException generata solo quando viene utilizzato il modello di aggiornamento batch di DAL. Il modello diretto del database non genera un'eccezione, ma indica semplicemente che non sono state interessate righe. Per illustrare questo problema, restituire gridView di entrambe le istanze del browser allo stato di pre-modifica. Quindi, nella prima istanza del browser fare clic sul pulsante Modifica e modificare il nome del prodotto da "Chai Tea" a "Chai" e fare clic su Aggiorna. Nella seconda finestra del browser fare clic sul pulsante Elimina per Chai.

Quando si fa clic su Elimina, la pagina esegue il postback, GridView richiama il metodo ObjectDataSource Delete() e ObjectDataSource chiama ProductsOptimisticConcurrencyBLL il metodo della DeleteProduct classe passando i valori originali. Il valore originale ProductName per la seconda istanza del browser è "Chai Tea", che non corrisponde al valore corrente ProductName nel database. Di conseguenza, l'istruzione DELETE rilasciata al database influisce su zero righe perché non è presente alcun record nel database che la WHERE clausola soddisfa. Il DeleteProduct metodo restituisce false e i dati di ObjectDataSource vengono rimbalzati in GridView.

Dal punto di vista dell'utente finale, facendo clic sul pulsante Elimina per Chai Tea nella seconda finestra del browser ha causato il flashing dello schermo e, al ritorno, il prodotto è ancora presente, anche se ora è elencato come "Chai" (la modifica del nome del prodotto apportata dalla prima istanza del browser). Se l'utente fa di nuovo clic sul pulsante Elimina, l'opzione Elimina avrà esito positivo, perché il valore originale ProductName di GridView ("Chai") corrisponde ora al valore nel database.

In entrambi questi casi, l'esperienza utente è lontana dall'ideale. È chiaro che non si vogliono mostrare all'utente i dettagli nitty-gritty dell'eccezione DBConcurrencyException quando si usa il modello di aggiornamento batch. E il comportamento quando si usa il modello diretto del database è un po 'confuso come il comando degli utenti non è riuscito, ma non c'era alcuna indicazione precisa del motivo.

Per risolvere questi due problemi, è possibile creare controlli Web etichetta nella pagina che forniscono una spiegazione del motivo per cui un aggiornamento o un'eliminazione non è riuscita. Per il modello di aggiornamento batch, è possibile determinare se si è verificata o meno un'eccezione DBConcurrencyException nel gestore eventi post-livello di GridView, visualizzando l'etichetta di avviso in base alle esigenze. Per il metodo diretto del database, è possibile esaminare il valore restituito del metodo BLL ,ovvero true se una riga è interessata, in caso contrario, false e visualizzare un messaggio informativo in base alle esigenze.

Passaggio 6: Aggiunta di messaggi informativi e visualizzazione di tali messaggi in caso di violazione della concorrenza

Quando si verifica una violazione della concorrenza, il comportamento esposto dipende dal fatto che sia stato usato il modello diretto di aggiornamento batch o database DAL. L'esercitazione usa entrambi i modelli, con il modello di aggiornamento batch usato per l'aggiornamento e il modello diretto del database usato per l'eliminazione. Per iniziare, aggiungere due controlli Web etichetta alla pagina che spiegano che si è verificata una violazione della concorrenza quando si tenta di eliminare o aggiornare i dati. Impostare le proprietà e del Visible controllo Etichetta su false. In questo modo le proprietà verranno nascoste in ogni pagina, ad eccezione di quelle specifiche visite di pagina in cui la proprietà Visible è impostata a livello di codice su true.EnableViewState

<asp:Label ID="DeleteConflictMessage" runat="server" Visible="False"
    EnableViewState="False" CssClass="Warning"
    Text="The record you attempted to delete has been modified by another user
           since you last visited this page. Your delete was cancelled to allow
           you to review the other user's changes and determine if you want to
           continue deleting this record." />
<asp:Label ID="UpdateConflictMessage" runat="server" Visible="False"
    EnableViewState="False" CssClass="Warning"
    Text="The record you attempted to update has been modified by another user
           since you started the update process. Your changes have been replaced
           with the current values. Please review the existing values and make
           any needed changes." />

Oltre a impostare Visiblele relative proprietà , EnabledViewStatee Text , ho anche impostato la CssClass proprietà su Warning, che fa sì che l'etichetta venga visualizzata in un tipo di carattere grande, rosso, corsivo, grassetto. Questa classe CSS Warning è stata definita e aggiunta a Styles.css di nuovo nell'esercitazione Analisi degli eventi associati all'inserimento, all'aggiornamento e all'eliminazione .

Dopo aver aggiunto queste etichette, il Designer in Visual Studio dovrebbe essere simile alla figura 18.

Alla pagina sono stati aggiunti due controlli etichetta

Figura 18: Sono stati aggiunti due controlli etichetta alla pagina (fare clic per visualizzare l'immagine a dimensione intera)

Con questi controlli Web Etichetta, è possibile esaminare come determinare quando si è verificata una violazione della concorrenza, a quel punto la proprietà dell'etichetta Visible appropriata può essere impostata su true, visualizzando il messaggio informativo.

Gestione delle violazioni di concorrenza durante l'aggiornamento

Si esamini prima di tutto come gestire le violazioni di concorrenza quando si usa il modello di aggiornamento batch. Poiché tali violazioni con il modello di aggiornamento batch causano la generazione di un'eccezione DBConcurrencyException , è necessario aggiungere codice alla pagina ASP.NET per determinare se si è verificata un'eccezione DBConcurrencyException durante il processo di aggiornamento. In tal caso, dovrebbe essere visualizzato un messaggio all'utente che spiega che le modifiche non sono state salvate perché un altro utente ha modificato gli stessi dati tra quando ha iniziato a modificare il record e quando ha fatto clic sul pulsante Aggiorna.

Come illustrato nell'esercitazione Sulla gestione delle eccezioni BLL e DAL-Level in un'esercitazione sulla pagina di ASP.NET , tali eccezioni possono essere rilevate e eliminate nei gestori eventi post-livello del controllo Web dei dati. È quindi necessario creare un gestore eventi per l'evento di RowUpdated GridView che controlla se è stata generata un'eccezione DBConcurrencyException . Questo gestore eventi viene passato un riferimento a qualsiasi eccezione generata durante il processo di aggiornamento, come illustrato nel codice del gestore eventi seguente:

protected void ProductsGrid_RowUpdated(object sender, GridViewUpdatedEventArgs e)
{
    if (e.Exception != null && e.Exception.InnerException != null)
    {
        if (e.Exception.InnerException is System.Data.DBConcurrencyException)
        {
            // Display the warning message and note that the
            // exception has been handled...
            UpdateConflictMessage.Visible = true;
            e.ExceptionHandled = true;
        }
    }
}

In caso di DBConcurrencyException eccezione, questo gestore eventi visualizza il UpdateConflictMessage controllo Etichetta e indica che l'eccezione è stata gestita. Con questo codice sul posto, quando si verifica una violazione della concorrenza durante l'aggiornamento di un record, le modifiche dell'utente andranno perse, poiché le modifiche di un altro utente sarebbero state sovrascritte contemporaneamente. In particolare, GridView viene restituito allo stato di pre-modifica e associato ai dati del database correnti. Verrà aggiornata la riga GridView con le modifiche dell'altro utente, che in precedenza non erano visibili. Inoltre, il UpdateConflictMessage controllo Etichetta spiega all'utente cosa è successo. Questa sequenza di eventi è dettagliata nella figura 19.

La Aggiornamenti di un utente si perde in faccia a una violazione di concorrenza

Figura 19: il Aggiornamenti di un utente viene perso in faccia a una violazione di concorrenza (fare clic per visualizzare l'immagine a dimensioni complete)

Nota

In alternativa, anziché restituire gridView allo stato di pre-modifica, è possibile lasciare GridView nello stato di modifica impostando la KeepInEditMode proprietà dell'oggetto passato GridViewUpdatedEventArgs su true. Se si accetta questo approccio, tuttavia, assicurarsi di ribintare i dati in GridView ( DataBind() richiamandone il metodo) in modo che i valori dell'altro utente vengano caricati nell'interfaccia di modifica. Il codice disponibile per il download con questa esercitazione include queste due righe di codice nel RowUpdated gestore eventi commentato. È sufficiente annullare ilcommentazione di queste righe di codice per avere GridView rimanere in modalità di modifica dopo una violazione della concorrenza.

Risposta alle violazioni di concorrenza durante l'eliminazione

Con il modello diretto del database, non esiste alcuna eccezione generata in caso di violazione di concorrenza. L'istruzione del database influisce semplicemente su nessun record, perché la clausola WHERE non corrisponde a alcun record. Tutti i metodi di modifica dei dati creati nel BLL sono stati progettati in modo che restituiscono un valore booleano che indica se hanno interessato esattamente un record. Pertanto, per determinare se si è verificata una violazione della concorrenza durante l'eliminazione DeleteProduct di un record, è possibile esaminare il valore restituito del metodo BLL.

Il valore restituito per un metodo BLL può essere esaminato nei gestori eventi post-livello di ObjectDataSource tramite la ReturnValue proprietà dell'oggetto ObjectDataSourceStatusEventArgs passato al gestore eventi. Poiché si è interessati a determinare il valore restituito dal DeleteProduct metodo, è necessario creare un gestore eventi per l'evento Deleted ObjectDataSource. La ReturnValue proprietà è di tipo object e può essere se null è stata generata un'eccezione e il metodo è stato interrotto prima di poter restituire un valore. Pertanto, è necessario assicurarsi prima di tutto che la ReturnValue proprietà non null sia e sia un valore booleano. Supponendo che questo controllo passi, viene visualizzato il DeleteConflictMessage controllo Etichetta se è falseReturnValue . Questa operazione può essere eseguita usando il codice seguente:

protected void ProductsOptimisticConcurrencyDataSource_Deleted(
    object sender, ObjectDataSourceStatusEventArgs e)
{
    if (e.ReturnValue != null && e.ReturnValue is bool)
    {
        bool deleteReturnValue = (bool)e.ReturnValue;
        if (deleteReturnValue == false)
        {
            // No row was deleted, display the warning message
            DeleteConflictMessage.Visible = true;
        }
    }
}

In caso di violazione della concorrenza, la richiesta di eliminazione dell'utente viene annullata. GridView viene aggiornato, che mostra le modifiche che si sono verificate per tale record tra il tempo in cui l'utente ha caricato la pagina e quando ha fatto clic sul pulsante Elimina. Quando si verifica una violazione di questo tipo, viene visualizzata l'etichetta DeleteConflictMessage , spiegando cosa è successo (vedere la figura 20).

L'eliminazione di un utente viene annullata in faccia a una violazione di concorrenza

Figura 20: l'eliminazione di un utente viene annullata in faccia a una violazione di concorrenza (fare clic per visualizzare l'immagine a dimensioni complete)

Riepilogo

Le opportunità per le violazioni di concorrenza esistono in ogni applicazione che consente a più utenti simultanei di aggiornare o eliminare i dati. Se tali violazioni non vengono rilevate, quando due utenti aggiornano simultaneamente gli stessi dati che ottengono nell'ultima scrittura "wins", sovrascrivendo le modifiche apportate all'altro utente. In alternativa, gli sviluppatori possono implementare un controllo di concorrenza ottimistico o pessimistico. Il controllo di concorrenza ottimistica presuppone che le violazioni della concorrenza siano infrequenti e non consentano semplicemente un comando di aggiornamento o eliminazione che costituisce una violazione della concorrenza. Il controllo di concorrenza pessimistico presuppone che le violazioni della concorrenza siano frequenti e semplicemente rifiutando il comando di aggiornamento o eliminazione di un utente non è accettabile. Con il controllo di concorrenza pessimistico, l'aggiornamento di un record comporta il blocco, impedendo così ad altri utenti di modificare o eliminare il record mentre è bloccato.

Typed DataSet in .NET offre funzionalità per supportare il controllo di concorrenza ottimistica. In particolare, le UPDATE istruzioni e DELETE rilasciate al database includono tutte le colonne della tabella, assicurandosi che l'aggiornamento o l'eliminazione si verifichino solo se i dati correnti del record corrispondono ai dati originali che l'utente aveva durante l'esecuzione dell'aggiornamento o dell'eliminazione. Dopo aver configurato DAL per supportare la concorrenza ottimistica, è necessario aggiornare i metodi BLL. Inoltre, la pagina di ASP.NET che chiama in base al BLL deve essere configurata in modo che ObjectDataSource recupera i valori originali dal controllo Web dei dati e li passa al BLL.

Come illustrato in questa esercitazione, l'implementazione del controllo di concorrenza ottimistica in un'applicazione Web di ASP.NET comporta l'aggiornamento di DAL e BLL e l'aggiunta del supporto nella pagina ASP.NET. Indipendentemente dal fatto che questo lavoro aggiunto sia un investimento saggio del tempo e dell'impegno dipende dall'applicazione. Se raramente si hanno utenti simultanei che aggiornano i dati o i dati che stanno aggiornando sono diversi tra loro, il controllo di concorrenza non è un problema chiave. Se, tuttavia, si hanno più utenti nel sito usando gli stessi dati, il controllo di concorrenza può aiutare a impedire agli aggiornamenti o alle eliminazioni di un utente di sovrascrivere inconsamente la sovrascrittura di un altro.

Programmazione felice!

Informazioni sull'autore

Scott Mitchell, autore di sette libri ASP/ASP.NET e fondatore di 4GuysFromRolla.com, ha lavorato con le tecnologie Microsoft Web dal 1998. Scott lavora come consulente indipendente, allenatore e scrittore. Il suo ultimo libro è Sams Teach Yourself ASP.NET 2,0 in 24 Ore. Può essere raggiunto a mitchell@4GuysFromRolla.com. o tramite il suo blog, che può essere trovato in http://ScottOnWriting.NET.