Erstellen einer Geschäftslogikebene (VB)
von Scott Mitchell
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.
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 DAL
BLL
. 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
, SuppliersBLL
und EmployeesBLL
.
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ückGetProductByProductID(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ückgibtAddProduct(productName, supplierID, categoryID, quantityPerUnit, unitPrice, unitsInStock, unitsOnOrder, reorderLevel, discontinued)
fügt ein neues Produkt mithilfe der übergebenen Werte in die Datenbank ein. gibt denProductID
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ückTrue
, wenn genau eine Zeile aktualisiert wurde,False
andernfallsDeleteProduct(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
, , GetProductsByCategoryID
und 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
, SupplierID
und 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 Nothing
soll. 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 DeleteProduct
ProductID
, 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 AddProduct
tun). 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 UpdateProduct
eine 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 SuppliersBLL
UpdateSupplierAddress
-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 SupplierDataTable
Methode "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.
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
ProductID
Felder ,ProductName
undDiscontinued
sind erforderlich, aber alle anderen Felder sind optional. - Die
UnitPrice
Felder ,UnitsInStock
,UnitsOnOrder
undReorderLevel
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 UnitPrice
Spalten , UnitsInStock
, UnitsOnOrder
oder 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.
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
.
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 UnitPrice
Spaltenwerte , UnitsInStock
UnitsOnOrder
, 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.