Freigeben über


Erstellen einer Geschäftslogikebene (VB)

von Scott Mitchell

PDF herunterladen

In diesem Tutorial erfahren Sie, wie Sie Ihre Geschäftsregeln in einer Business Logic Layer (BLL) zentralisieren, die als Vermittler für den Datenaustausch zwischen der Präsentationsebene und der DAL dient.

Einführung

Die im ersten Tutorial erstellte Datenzugriffsebene (Data Access Layer, DAL) trennt die Datenzugriffslogik sauber von der Präsentationslogik. Obwohl die DAL die Datenzugriffsdetails sauber von der Präsentationsebene trennt, erzwingt sie keine möglicherweise zutreffenden Geschäftsregeln. Für unsere Anwendung möchten wir beispielsweise zulassen, dass die CategoryID Felder oder SupplierID der Products Tabelle geändert werden können, wenn das Discontinued Feld auf 1 festgelegt ist, oder wir möchten Seniority-Regeln erzwingen, indem wir Situationen verbieten, in denen ein Mitarbeiter von einer Person verwaltet wird, die nach ihm eingestellt wurde. Ein weiteres häufiges Szenario ist die Autorisierung, möglicherweise können nur Benutzer in einer bestimmten Rolle Produkte löschen oder den UnitPrice Wert ändern.

In diesem Tutorial erfahren Sie, wie Sie diese Geschäftsregeln in einer Business Logic Layer (BLL) zentralisieren, die als Vermittler für den Datenaustausch zwischen der Präsentationsebene und der DAL dient. In einer realen Anwendung sollte die BLL als separates Klassenbibliotheksprojekt implementiert werden. Für diese Tutorials implementieren wir die BLL jedoch als eine Reihe von Klassen in unserem App_Code Ordner, um die Projektstruktur zu vereinfachen. Abbildung 1 veranschaulicht die architektonischen Beziehungen zwischen der Präsentationsebene, BLL und DAL.

Die BLL trennt die Präsentationsebene von der Datenzugriffsebene und erzwingt Geschäftsregeln.

Abbildung 1: Die BLL trennt die Präsentationsebene von der Datenzugriffsebene und erzwingt Geschäftsregeln.

Anstatt separate Klassen zu erstellen, um unsere Geschäftslogik zu implementieren, können wir diese Logik alternativ direkt im Typisierten DataSet mit partiellen Klassen platzieren. Ein Beispiel zum Erstellen und Erweitern eines typisierten DataSets finden Sie im ersten Tutorial.

Schritt 1: Erstellen der BLL-Klassen

Unser BLL besteht aus vier Klassen, eine für jeden TableAdapter im DAL; jede dieser BLL-Klassen verfügt über Methoden zum Abrufen, Einfügen, Aktualisieren und Löschen aus dem jeweiligen TableAdapter in der DAL, wobei die entsprechenden Geschäftsregeln angewendet werden.

Um die DAL- und BLL-bezogenen Klassen sauberer zu trennen, erstellen wir zwei Unterordner im App_Code Ordner und DALBLL. Klicken Sie einfach mit der rechten Maustaste auf den App_Code Ordner im Projektmappen-Explorer, und wählen Sie Neuer Ordner aus. Verschieben Sie nach dem Erstellen dieser beiden Ordner das typisierte DataSet, das im ersten Tutorial erstellt wurde, in den DAL Unterordner.

Erstellen Sie als Nächstes die vier BLL-Klassendateien im BLL Unterordner. Klicken Sie dazu mit der rechten Maustaste auf den BLL Unterordner, wählen Sie Neues Element hinzufügen und dann die Klasse-Vorlage aus. Nennen Sie die vier Klassen ProductsBLL, CategoriesBLL, SuppliersBLLund EmployeesBLL.

Hinzufügen von vier neuen Klassen zum App_Code Ordner

Abbildung 2: Hinzufügen von vier neuen Klassen zum App_Code Ordner

Als Nächstes fügen wir jeder Klasse Methoden hinzu, um einfach die Methoden zu umschließen, die für die TableAdapters aus dem ersten Tutorial definiert wurden. Vorerst rufen diese Methoden nur direkt in die DAL auf. Wir kehren später zurück, um die erforderliche Geschäftslogik hinzuzufügen.

Hinweis

Wenn Sie Visual Studio Standard Edition oder höher verwenden (d. h., Sie verwenden nicht Visual Web Developer), können Sie Ihre Klassen optional mithilfe der Designer Klasse visuell entwerfen. Weitere Informationen zu diesem neuen Feature in Visual Studio finden Sie im Blog "Class Designer".

Für die ProductsBLL -Klasse müssen insgesamt sieben Methoden hinzugefügt werden:

  • GetProducts() gibt alle Produkte zurück
  • GetProductByProductID(productID) gibt das Produkt mit der angegebenen Produkt-ID zurück.
  • GetProductsByCategoryID(categoryID) gibt alle Produkte aus der angegebenen Kategorie zurück.
  • GetProductsBySupplier(supplierID) alle Produkte des angegebenen Lieferanten zurückgibt
  • AddProduct(productName, supplierID, categoryID, quantityPerUnit, unitPrice, unitsInStock, unitsOnOrder, reorderLevel, discontinued) fügt ein neues Produkt mithilfe der übergebenen Werte in die Datenbank ein. gibt den ProductID Wert des neu eingefügten Datensatzes zurück.
  • UpdateProduct(productName, supplierID, categoryID, quantityPerUnit, unitPrice, unitsInStock, unitsOnOrder, reorderLevel, discontinued, productID) aktualisiert ein vorhandenes Produkt in der Datenbank mithilfe der übergebenen Werte; gibt zurück True , wenn genau eine Zeile aktualisiert wurde, False andernfalls
  • DeleteProduct(productID) löscht das angegebene Produkt aus der Datenbank.

ProductsBLL.vb

Imports NorthwindTableAdapters

<System.ComponentModel.DataObject()> _
Public Class ProductsBLL

    Private _productsAdapter As ProductsTableAdapter = Nothing
    Protected ReadOnly Property Adapter() As ProductsTableAdapter
        Get
            If _productsAdapter Is Nothing Then
                _productsAdapter = New ProductsTableAdapter()
            End If

            Return _productsAdapter
        End Get
    End Property

    <System.ComponentModel.DataObjectMethodAttribute _
        (System.ComponentModel.DataObjectMethodType.Select, True)> _
    Public Function GetProducts() As Northwind.ProductsDataTable
        Return Adapter.GetProducts()
    End Function

    <System.ComponentModel.DataObjectMethodAttribute _
        (System.ComponentModel.DataObjectMethodType.Select, False)> _
    Public Function GetProductByProductID(ByVal productID As Integer) _
        As Northwind.ProductsDataTable
        Return Adapter.GetProductByProductID(productID)
    End Function

    <System.ComponentModel.DataObjectMethodAttribute _
        (System.ComponentModel.DataObjectMethodType.Select, False)> _
    Public Function GetProductsByCategoryID(ByVal categoryID As Integer) _
        As Northwind.ProductsDataTable
        Return Adapter.GetProductsByCategoryID(categoryID)
    End Function

    <System.ComponentModel.DataObjectMethodAttribute _
        (System.ComponentModel.DataObjectMethodType.Select, False)> _
    Public Function GetProductsBySupplierID(ByVal supplierID As Integer) _
        As Northwind.ProductsDataTable
        Return Adapter.GetProductsBySupplierID(supplierID)
    End Function

    <System.ComponentModel.DataObjectMethodAttribute _
        (System.ComponentModel.DataObjectMethodType.Insert, True)> _
    Public Function AddProduct( _
        productName As String, supplierID As Nullable(Of Integer), _
        categoryID As Nullable(Of Integer), quantityPerUnit As String, _
        unitPrice As Nullable(Of Decimal), unitsInStock As Nullable(Of Short), _
        unitsOnOrder As Nullable(Of Short), reorderLevel As Nullable(Of Short), _
        discontinued As Boolean) _
        As Boolean

        Dim products As New Northwind.ProductsDataTable()
        Dim product As Northwind.ProductsRow = products.NewProductsRow()

        product.ProductName = productName
        If Not supplierID.HasValue Then
            product.SetSupplierIDNull()
        Else
            product.SupplierID = supplierID.Value
        End If

        If Not categoryID.HasValue Then
            product.SetCategoryIDNull()
        Else
            product.CategoryID = categoryID.Value
        End If

        If quantityPerUnit Is Nothing Then
            product.SetQuantityPerUnitNull()
        Else
            product.QuantityPerUnit = quantityPerUnit
        End If

        If Not unitPrice.HasValue Then
            product.SetUnitPriceNull()
        Else
            product.UnitPrice = unitPrice.Value
        End If

        If Not unitsInStock.HasValue Then
            product.SetUnitsInStockNull()
        Else
            product.UnitsInStock = unitsInStock.Value
        End If

        If Not unitsOnOrder.HasValue Then
            product.SetUnitsOnOrderNull()
        Else
            product.UnitsOnOrder = unitsOnOrder.Value
        End If

        If Not reorderLevel.HasValue Then
            product.SetReorderLevelNull()
        Else
            product.ReorderLevel = reorderLevel.Value
        End If

        product.Discontinued = discontinued

        products.AddProductsRow(product)
        Dim rowsAffected As Integer = Adapter.Update(products)

        Return rowsAffected = 1
    End Function

    <System.ComponentModel.DataObjectMethodAttribute _
        (System.ComponentModel.DataObjectMethodType.Update, True)> _
    Public Function UpdateProduct(_
        productName As String, supplierID As Nullable(Of Integer), _
        categoryID As Nullable(Of Integer), quantityPerUnit As String, _
        unitPrice As Nullable(Of Decimal), unitsInStock As Nullable(Of Short), _
        unitsOnOrder As Nullable(Of Short), reorderLevel As Nullable(Of Short), _
        discontinued As Boolean, productID As Integer) _
        As Boolean

        Dim products As Northwind.ProductsDataTable = _
            Adapter.GetProductByProductID(productID)

        If products.Count = 0 Then
            Return False
        End If

        Dim product as Northwind.ProductsRow = products(0)

        product.ProductName = productName
        If Not supplierID.HasValue Then
            product.SetSupplierIDNull()
        Else
            product.SupplierID = supplierID.Value
        End If

        If Not categoryID.HasValue Then
            product.SetCategoryIDNull()
        Else
            product.CategoryID = categoryID.Value
        End If

        If quantityPerUnit Is Nothing Then
            product.SetQuantityPerUnitNull()
        Else
            product.QuantityPerUnit = quantityPerUnit
        End If

        If Not unitPrice.HasValue Then
            product.SetUnitPriceNull()
        Else
            product.UnitPrice = unitPrice.Value
        End If

        If Not unitsInStock.HasValue Then
            product.SetUnitsInStockNull()
        Else
            product.UnitsInStock = unitsInStock.Value
        End If

        If Not unitsOnOrder.HasValue Then
            product.SetUnitsOnOrderNull()
        Else
            product.UnitsOnOrder = unitsOnOrder.Value
        End If

        If Not reorderLevel.HasValue Then
            product.SetReorderLevelNull()
        Else
            product.ReorderLevel = reorderLevel.Value
        End If

        product.Discontinued = discontinued

        Dim rowsAffected As Integer = Adapter.Update(product)

        Return rowsAffected = 1
    End Function

    <System.ComponentModel.DataObjectMethodAttribute _
        (System.ComponentModel.DataObjectMethodType.Delete, True)> _
    Public Function DeleteProduct(ByVal productID As Integer) As Boolean
        Dim rowsAffected As Integer = Adapter.Delete(productID)

        Return rowsAffected = 1
    End Function
End Class

Die Methoden, die einfach Daten GetProducts, GetProductByProductID, , GetProductsByCategoryIDund GetProductBySuppliersID zurückgeben, sind ziemlich einfach, da sie einfach in die DAL aufrufen. Während es in einigen Szenarien Geschäftsregeln geben kann, die auf dieser Ebene implementiert werden müssen (z. B. Autorisierungsregeln, die auf dem aktuell angemeldeten Benutzer oder der Rolle basieren, zu der der Benutzer gehört), lassen wir diese Methoden einfach unverändert. Für diese Methoden dient die BLL dann lediglich als Proxy, über den die Präsentationsebene auf die zugrunde liegenden Daten der Datenzugriffsebene zugreift.

Die AddProduct Methoden und UpdateProduct übernehmen als Parameter die Werte für die verschiedenen Produktfelder und fügen ein neues Produkt bzw. aktualisieren ein vorhandenes Produkt. Da viele Spalten der Product Tabelle Werte akzeptieren NULL können (CategoryID, SupplierIDund UnitPrice, um nur einige zu nennen), verwenden diese Eingabeparameter für AddProduct und UpdateProduct , die diesen Spalten zugeordnet sind , NULLable-Typen. Nullable-Typen sind neu in .NET 2.0 und bieten eine Technik zum Angeben, ob ein Werttyp stattdessen sein Nothingsoll. Weitere Informationen finden Sie im Blogeintrag The Truth About Nullable Types and VB und in der technischen Dokumentation zur Nullable-Struktur von Paul Vick.

Alle drei Methoden geben einen booleschen Wert zurück, der angibt, ob eine Zeile eingefügt, aktualisiert oder gelöscht wurde, da der Vorgang möglicherweise nicht zu einer betroffenen Zeile führt. Wenn der Seitenentwickler beispielsweise für ein nicht vorhandenes Produkt die Übergabe aufruft DeleteProductProductID , hat die DELETE an die Datenbank ausgegebene Anweisung keine Auswirkungen, und daher gibt die DeleteProduct Methode zurück False.

Beachten Sie, dass beim Hinzufügen eines neuen Produkts oder beim Aktualisieren eines vorhandenen Produkts die Feldwerte des neuen oder geänderten Produkts als Liste von Skalaren verwendet werden, anstatt ein ProductsRow instance zu akzeptieren. Dieser Ansatz wurde gewählt, weil die ProductsRow -Klasse von der ADO.NET-Klasse DataRow abgeleitet ist, die keinen standardparameterlosen Konstruktor aufweist. Um eine neue ProductsRow instance zu erstellen, müssen wir zuerst eine ProductsDataTable instance erstellen und dann dessen NewProductRow() Methode aufrufen (was wir in AddProducttun). Dieser Mangel rückt den Kopf zurück, wenn wir uns mit dem Einfügen und Aktualisieren von Produkten mit ObjectDataSource zuwenden. Kurz gesagt, die ObjectDataSource versucht, eine instance der Eingabeparameter zu erstellen. Wenn die BLL-Methode eine ProductsRow instance erwartet, versucht die ObjectDataSource, eine zu erstellen, schlägt jedoch aufgrund des Fehlens eines parameterlosen Standardkonstruktors fehl. Weitere Informationen zu diesem Problem finden Sie in den folgenden beiden ASP.NET Forenbeiträgen: Aktualisieren von ObjectDataSources mit Strongly-Typed DataSets und Problem mit ObjectDataSource und Strongly-Typed DataSet.

Als Nächstes erstellt der Code in AddProduct und UpdateProducteine ProductsRow instance und füllt ihn mit den gerade übergebenen Werten auf. Beim Zuweisen von Werten zu DataColumns eines DataRow können verschiedene Überprüfungen auf Feldebene durchgeführt werden. Das manuelle Einfügen der übergebenen Werte in einen DataRow-Code trägt daher dazu bei, die Gültigkeit der Daten, die an die BLL-Methode übergeben werden, sicherzustellen. Leider verwenden die stark typisierten DataRow-Klassen, die von Visual Studio generiert werden, keine NULLable-Typen. Um stattdessen anzugeben, dass ein bestimmter DataColumn in einem DataRow einem NULL Datenbankwert entsprechen soll, müssen wir die SetColumnNameNull() -Methode verwenden.

In UpdateProduct laden wir zuerst in das Produkt, um mit zu aktualisieren GetProductByProductID(productID). Obwohl dies wie eine unnötige Reise in die Datenbank erscheinen mag, wird sich diese zusätzliche Reise in zukünftigen Tutorials als lohnenswert erweisen, in denen die optimistische Parallelität untersucht wird. Die optimistische Parallelität ist eine Technik, mit der sichergestellt wird, dass zwei Benutzer, die gleichzeitig an den gleichen Daten arbeiten, die Änderungen des anderen nicht versehentlich überschreiben. Durch das Abrufen des gesamten Datensatzes wird es auch einfacher, Updatemethoden in der BLL zu erstellen, die nur eine Teilmenge der DataRow-Spalten ändern. Wenn wir die SuppliersBLL Klasse untersuchen, sehen wir ein solches Beispiel.

Beachten Sie schließlich, dass für die ProductsBLL Klasse das DataObject-Attribut angewendet wird (die [System.ComponentModel.DataObject] Syntax direkt vor der Klassenanweisung am Anfang der Datei) und die Methoden dataObjectMethodAttribute-Attribute aufweisen. Das DataObject -Attribut kennzeichnet die -Klasse als ein Objekt, das für die Bindung an ein ObjectDataSource-Steuerelement geeignet ist, während der den DataObjectMethodAttribute Zweck der Methode angibt. Wie wir in zukünftigen Tutorials sehen werden, erleichtert die ObjectDataSource von ASP.NET 2.0 den deklarativen Zugriff auf Daten aus einer Klasse. Um die Liste der möglichen Klassen zu filtern, an die im ObjectDataSource-Assistenten gebunden werden soll, werden standardmäßig nur die Klassen markiert, die in DataObjects der Dropdownliste des Assistenten angezeigt werden. Die ProductsBLL -Klasse funktioniert auch ohne diese Attribute, aber das Hinzufügen dieser Attribute erleichtert die Arbeit im ObjectDataSource-Assistenten.

Hinzufügen der anderen Klassen

Wenn die ProductsBLL Klasse abgeschlossen ist, müssen wir weiterhin die Klassen für die Arbeit mit Kategorien, Lieferanten und Mitarbeitern hinzufügen. Nehmen Sie sich einen Moment Zeit, um die folgenden Klassen und Methoden mithilfe der Konzepte aus dem obigen Beispiel zu erstellen:

  • CategoriesBLL.cs

    • GetCategories()
    • GetCategoryByCategoryID(categoryID)
  • SuppliersBLL.cs

    • GetSuppliers()
    • GetSupplierBySupplierID(supplierID)
    • GetSuppliersByCountry(country)
    • UpdateSupplierAddress(supplierID, address, city, country)
  • EmployeesBLL.cs

    • GetEmployees()
    • GetEmployeeByEmployeeID(employeeID)
    • GetEmployeesByManager(managerID)

Die einzige Methode, die erwähnenswert ist, ist die -Methode der SuppliersBLLUpdateSupplierAddress -Klasse. Diese Methode bietet eine Schnittstelle zum Aktualisieren nur der Adressinformationen des Lieferanten. Intern liest diese Methode das -Objekt für das SupplierDataRow angegebene supplierID (mithilfe von GetSupplierBySupplierID), legt die adressbezogenen Eigenschaften fest und ruft dann die SupplierDataTableMethode "s Update " auf. Die UpdateSupplierAddress Methode folgt:

<System.ComponentModel.DataObjectMethodAttribute _
    (System.ComponentModel.DataObjectMethodType.Update, True)> _
Public Function UpdateSupplierAddress(ByVal supplierID As Integer, _
    ByVal address As String, ByVal city As String, ByVal country As String) _
    As Boolean

    Dim suppliers As Northwind.SuppliersDataTable = _
        Adapter.GetSupplierBySupplierID(supplierID)

    If suppliers.Count = 0 Then
        Return False
    Else
        Dim supplier As Northwind.SuppliersRow = suppliers(0)

        If address Is Nothing Then
            supplier.SetAddressNull()
        Else
            supplier.Address = address
        End If

        If city Is Nothing Then
            supplier.SetCityNull()
        Else
            supplier.City = city
        End If

        If country Is Nothing Then
            supplier.SetCountryNull()
        Else
            supplier.Country = country
        End If

        Dim rowsAffected As Integer = Adapter.Update(supplier)

        Return rowsAffected = 1
    End If
End Function

Meine vollständige Implementierung der BLL-Klassen finden Sie im Download dieses Artikels.

Schritt 2: Zugreifen auf die typisierten DataSets über die BLL-Klassen

Im ersten Tutorial haben wir Beispiele für die programmgesteuerte direkte Arbeit mit dem Typisiertes DataSet gesehen, aber mit dem Hinzufügen unserer BLL-Klassen sollte die Präsentationsebene stattdessen für die BLL funktionieren. Im Beispiel aus AllProducts.aspx dem ersten Tutorial wurde verwendet ProductsTableAdapter , um die Liste der Produkte an eine GridView zu binden, wie im folgenden Code gezeigt:

Dim productsAdapter As New ProductsTableAdapter()
GridView1.DataSource = productsAdapter.GetProducts()
GridView1.DataBind()

Um die neuen BLL-Klassen zu verwenden, muss lediglich die erste Codezeile geändert werden, indem Sie einfach das ProductsTableAdapter Objekt durch ein ProductBLL -Objekt ersetzen:

Dim productLogic As New ProductsBLL()
GridView1.DataSource = productLogic.GetProducts()
GridView1.DataBind()

Auf die BLL-Klassen kann auch deklarativ zugegriffen werden (wie auch auf das Typisierte DataSet) mithilfe der ObjectDataSource. In den folgenden Tutorials wird die ObjectDataSource ausführlicher erläutert.

Die Liste der Produkte wird in einer GridView angezeigt.

Abbildung 3: Die Liste der Produkte wird in einer GridView angezeigt (Klicken Sie hier, um das bild in voller Größe anzuzeigen)

Schritt 3: Hinzufügen Field-Level Validierung zu den DataRow-Klassen

Bei der Überprüfung auf Feldebene handelt es sich um Überprüfungen, die sich auf die Eigenschaftswerte der Geschäftsobjekte beim Einfügen oder Aktualisieren beziehen. Zu den Validierungsregeln auf Feldebene für Produkte gehören:

  • Das ProductName Feld muss mindestens 40 Zeichen lang sein.
  • Das QuantityPerUnit Feld muss mindestens 20 Zeichen lang sein.
  • Die ProductIDFelder , ProductNameund Discontinued sind erforderlich, aber alle anderen Felder sind optional.
  • Die UnitPriceFelder , UnitsInStock, UnitsOnOrderund ReorderLevel müssen größer oder gleich 0 sein.

Diese Regeln können und sollten auf Datenbankebene ausgedrückt werden. Das Zeichenlimit für die ProductName Felder und QuantityPerUnit wird von den Datentypen dieser Spalten in der Products Tabelle (nvarchar(40) bzw nvarchar(20). ) erfasst. Gibt an, ob Felder erforderlich und optional sind, wenn die Datenbanktabellenspalte s zulässt NULL . Es gibt vier Prüfeinschränkungen , die sicherstellen, dass nur Werte, die größer oder gleich 0 sind, in die UnitPriceSpalten , UnitsInStock, UnitsOnOrderoder ReorderLevel gelangen können.

Zusätzlich zum Erzwingen dieser Regeln in der Datenbank sollten sie auch auf der DataSet-Ebene erzwungen werden. Tatsächlich werden die Feldlänge und die Angabe, ob ein Wert erforderlich oder optional ist, bereits für jeden DataTable-Satz von DataColumns erfasst. Um die vorhandene Überprüfung auf Feldebene automatisch anzuzeigen, wechseln Sie zum dataSet-Designer, wählen Sie ein Feld aus einer der DataTables aus, und wechseln Sie dann zum Eigenschaftenfenster. Wie abbildung 4 zeigt, hat die QuantityPerUnit DataColumn im ProductsDataTable eine maximale Länge von 20 Zeichen und lässt Werte zu NULL . Wenn wir versuchen, die ProductsDataRow-Eigenschaft von QuantityPerUnit auf einen Zeichenfolgenwert festzulegen, der länger als 20 Zeichen ist, wird eine ArgumentException ausgelöst.

DataColumn bietet grundlegende Field-Level Validierung

Abbildung 4: DataColumn bietet grundlegende Field-Level Validierung (Klicken Sie hier, um das Bild in voller Größe anzuzeigen)

Leider können wir keine Begrenzungsüberprüfungen angeben, z. B. dass der UnitPrice Wert größer als oder gleich 0 sein muss, über die Eigenschaftenfenster. Um diese Art der Validierung auf Feldebene bereitzustellen, müssen wir einen Ereignishandler für das ColumnChanging-Ereignis der DataTable erstellen. Wie im vorherigen Tutorial erwähnt, können die vom typisierten DataSet erstellten DataSet-, DataTables- und DataRow-Objekte mithilfe von partiellen Klassen erweitert werden. Mit dieser Technik können wir einen ColumnChanging Ereignishandler für die ProductsDataTable -Klasse erstellen. Erstellen Sie zunächst eine Klasse im Ordner mit dem App_Code Namen ProductsDataTable.ColumnChanging.vb.

Hinzufügen einer neuen Klasse zum ordner

Abbildung 5: Hinzufügen einer neuen Klasse zum App_Code Ordner (Klicken Sie hier, um das Bild in voller Größe anzuzeigen)

Erstellen Sie als Nächstes einen Ereignishandler für das ColumnChanging Ereignis, der sicherstellt, dass die UnitPriceSpaltenwerte , UnitsInStockUnitsOnOrder, und ReorderLevel (falls nicht NULL) größer oder gleich Null sind. Wenn eine solche Spalte außerhalb des Bereichs liegt, lösen Sie eine aus ArgumentException.

ProductsDataTable.ColumnChanging.vb

Imports System.data

Partial Public Class Northwind
    Partial Public Class ProductsDataTable
        Public Overrides Sub BeginInit()
            AddHandler Me.ColumnChanging, AddressOf ValidateColumn
        End Sub

        Sub ValidateColumn(sender As Object, e As DataColumnChangeEventArgs)
            If e.Column.Equals(Me.UnitPriceColumn) Then
                If Not Convert.IsDBNull(e.ProposedValue) AndAlso _
                    CType(e.ProposedValue, Decimal) < 0 Then
                    Throw New ArgumentException( _
                        "UnitPrice cannot be less than zero", "UnitPrice")
                End If
            ElseIf e.Column.Equals(Me.UnitsInStockColumn) OrElse _
                e.Column.Equals(Me.UnitsOnOrderColumn) OrElse _
                e.Column.Equals(Me.ReorderLevelColumn) Then
                If Not Convert.IsDBNull(e.ProposedValue) AndAlso _
                    CType(e.ProposedValue, Short) < 0 Then
                    Throw New ArgumentException(String.Format( _
                        "{0} cannot be less than zero", e.Column.ColumnName), _
                        e.Column.ColumnName)
                End If
            End If
        End Sub
    End Class
End Class

Schritt 4: Hinzufügen von benutzerdefinierten Geschäftsregeln zu den Klassen der BLL

Zusätzlich zur Validierung auf Feldebene können allgemeine benutzerdefinierte Geschäftsregeln vorhanden sein, die verschiedene Entitäten oder Konzepte umfassen, die auf der Ebene einzelner Spalten nicht ausgedrückt werden können, z. B.:

  • Wenn ein Produkt eingestellt wird, kann es UnitPrice nicht aktualisiert werden.
  • Das Wohnsitzland eines Mitarbeiters muss mit dem Wohnsitzland des Vorgesetzten übereinstimmen.
  • Ein Produkt kann nicht eingestellt werden, wenn es sich um das einzige produkt handelt, das vom Lieferanten bereitgestellt wird

Die BLL-Klassen sollten Überprüfungen enthalten, um die Einhaltung der Geschäftsregeln der Anwendung sicherzustellen. Diese Überprüfungen können den Methoden, auf die sie angewendet werden, direkt hinzugefügt werden.

Stellen Sie sich vor, dass unsere Geschäftsregeln vorschreiben, dass ein Produkt nicht als eingestellt gekennzeichnet werden kann, wenn es das einzige Produkt eines bestimmten Lieferanten ist. Das heißt, wenn Produkt X das einzige Produkt war, das wir beim Lieferanten Y gekauft haben, könnten wir X nicht als nicht mehr eingestellt kennzeichnen; Wenn uns der Lieferant Y jedoch drei Produkte geliefert hat, A, B und C, dann könnten wir alle als nicht mehr verfügbar kennzeichnen. Eine ungewöhnliche Geschäftsregel, aber Geschäftsregeln und gesunder Menschenverstand sind nicht immer aufeinander abgestimmt!

Um diese Geschäftsregel in der UpdateProducts -Methode zu erzwingen, würden wir zunächst überprüfen, ob Discontinued auf True festgelegt wurde, und wenn ja, rufen wir auf GetProductsBySupplierID , um zu ermitteln, wie viele Produkte wir vom Lieferanten dieses Produkts erworben haben. Wenn nur ein Produkt von diesem Lieferanten gekauft wird, lösen wir eine aus ApplicationException.

<System.ComponentModel.DataObjectMethodAttribute_
    (System.ComponentModel.DataObjectMethodType.Update, True)> _
Public Function UpdateProduct( _
    productName As String, supplierID As Nullable(Of Integer), _
    categoryID As Nullable(Of Integer), quantityPerUnit As String, _
    unitPrice As Nullable(Of Decimal), unitsInStock As Nullable(Of Short), _
    unitsOnOrder As Nullable(Of Short), reorderLevel As Nullable(Of Short), _
    discontinued As Boolean, productID As Integer) _
    As Boolean

    Dim products As Northwind.ProductsDataTable = _
        Adapter.GetProductByProductID(productID)

    If products.Count = 0 Then
        Return False
    End If

    Dim product As Northwind.ProductsRow = products(0)

    If discontinued Then
        Dim productsBySupplier As Northwind.ProductsDataTable = _
            Adapter.GetProductsBySupplierID(product.SupplierID)

        If productsBySupplier.Count = 1 Then
            Throw New ApplicationException( _
                "You cannot mark a product as discontinued if it is " & _
                "the only product purchased from a supplier")
        End If
    End If

    product.ProductName = productName

    If Not supplierID.HasValue Then
        product.SetSupplierIDNull()
    Else
        product.SupplierID = supplierID.Value
    End If

    If Not categoryID.HasValue Then
        product.SetCategoryIDNull()
    Else
        product.CategoryID = categoryID.Value
    End If

    If quantityPerUnit Is Nothing Then
        product.SetQuantityPerUnitNull()
    Else
        product.QuantityPerUnit = quantityPerUnit
    End If

    If Not unitPrice.HasValue Then
        product.SetUnitPriceNull()
    Else
        product.UnitPrice = unitPrice.Value
    End If

    If Not unitsInStock.HasValue Then
        product.SetUnitsInStockNull()
    Else
        product.UnitsInStock = unitsInStock.Value
    End If

    If Not unitsOnOrder.HasValue Then
        product.SetUnitsOnOrderNull()
    Else
        product.UnitsOnOrder = unitsOnOrder.Value
    End If

    If Not reorderLevel.HasValue Then
        product.SetReorderLevelNull()
    Else
        product.ReorderLevel = reorderLevel.Value
    End If

    product.Discontinued = discontinued

    Dim rowsAffected As Integer = Adapter.Update(product)

    Return rowsAffected = 1
End Function

Reagieren auf Validierungsfehler in der Präsentationsebene

Beim Aufrufen der BLL von der Präsentationsebene können wir entscheiden, ob sie versuchen, ausnahmen zu behandeln, die ausgelöst werden können, oder ob sie bis zu ASP.NET blasen (wodurch das -Ereignis des HttpApplication-Ereignisses Error ausgelöst wird). Um eine Ausnahme beim programmgesteuerten Arbeiten mit der BLL zu behandeln, können wir einen Try... Catch-Block , wie im folgenden Beispiel gezeigt:

Dim productLogic As New ProductsBLL()

Try
    productLogic.UpdateProduct("Scotts Tea", 1, 1, Nothing, _
      -14, 10, Nothing, Nothing, False, 1)
Catch ae As ArgumentException
    Response.Write("There was a problem: " & ae.Message)
End Try

Wie wir in zukünftigen Tutorials sehen werden, kann die Behandlung von Ausnahmen, die von der BLL bei Verwendung eines Datenwebsteuerelements zum Einfügen, Aktualisieren oder Löschen von Daten verwendet werden, direkt in einem Ereignishandler behandelt werden, anstatt Code in Try...Catch Blöcken umschließen zu müssen.

Zusammenfassung

Eine gut gestaltete Anwendung wird in unterschiedliche Ebenen erstellt, von denen jede eine bestimmte Rolle kapselt. Im ersten Tutorial dieser Artikelreihe haben wir eine Datenzugriffsebene mit typisierten DataSets erstellt. In diesem Tutorial haben wir eine Geschäftslogikebene als eine Reihe von Klassen im Ordner unserer Anwendung App_Code erstellt, die unsere DAL aufrufen. Die BLL implementiert die Logik auf Feld- und Geschäftsebene für unsere Anwendung. Zusätzlich zum Erstellen einer separaten BLL, wie in diesem Tutorial, besteht eine weitere Möglichkeit darin, die TableAdapters-Methoden durch die Verwendung partieller Klassen zu erweitern. Die Verwendung dieser Technik ermöglicht es uns jedoch nicht, vorhandene Methoden zu überschreiben und unseren DAL und unseren BLL genauso sauber zu trennen wie der in diesem Artikel beschriebene Ansatz.

Nachdem DAL und BLL abgeschlossen sind, können wir mit der Präsentationsebene beginnen. Im nächsten Tutorial machen wir einen kurzen Umweg zu Den Themen zum Datenzugriff und definieren ein konsistentes Seitenlayout für die Verwendung in den Tutorials.

Viel Spaß beim Programmieren!

Zum Autor

Scott Mitchell, Autor von sieben ASP/ASP.NET-Büchern und Gründer von 4GuysFromRolla.com, arbeitet seit 1998 mit Microsoft-Webtechnologien. Scott arbeitet als unabhängiger Berater, Trainer und Autor. Sein neuestes Buch ist Sams Teach Yourself ASP.NET 2.0 in 24 Hours. Er kann unter mitchell@4GuysFromRolla.comoder über seinen Blog erreicht werden, der unter http://ScottOnWriting.NETzu finden ist.

Besonderer Dank an

Diese Tutorialreihe wurde von vielen hilfreichen Prüfern überprüft. Hauptprüfer für dieses Tutorial waren Liz Shulok, Dennis Patterson, Carlos Santos und Hilton Giesenow. Möchten Sie meine bevorstehenden MSDN-Artikel lesen? Wenn dies der Fall ist, legen Sie eine Zeile unter abmitchell@4GuysFromRolla.com.