Uno dei grandi vantaggi dei code contracts di Microsoft (vedere gli articoli sul sito www.domusdotnet.org per maggiori info) è quello di poter definire i contratti anche sulle interfacce.
Ad esempio le precondizioni si definiscono attraverso uno dei metodi statici Requires della classe Contracts (System.Diagnostic.Contracts):
- Imports System.Diagnostics.Contracts
- Module Module1
- Sub Main()
- Dim chr = GetLastChar("pippo")
- Dim chr2 = GetLastChar("")
- End Sub
- Public Function GetLastChar(str As String) As String
- Contract.Requires(Not String.IsNullOrEmpty(str))
- Return str.Substring(str.Length - 1)
- End Function
- End Module
Il metodo GetLastChar richiede che l’argomento non sia né Nothing, né “” e, nel caso precedente, in fase di compilazione, si ottiene il seguente warning
mentre in fase di esecuzione si ottiene la seguente eccezione:
Fino a qui tutto bene, ma come facciamo a definire le precondizioni (ma il concetto vale anche per postcondizioni ed invarianti d’oggetto) nel caso di interfacce o di metodi astratti di classi in cui non abbiamo la possibilità di inserire del codice?
In questo caso abbiamo la possibilità di definire una classe che funge da “contenitore” per le definizioni dei contratti. Ad esempio:
- <ContractClass(GetType(InterfacciaContract))>
- Public Interface IInterfaccia
- Function Metodo(arg As Integer) As Integer
- End Interface
- <ContractClassFor(GetType(IInterfaccia))>
- Public MustInherit Class InterfacciaContract
- Implements IInterfaccia
- Public Function Metodo(arg As Integer) As Integer Implements IInterfaccia.Metodo
- Contract.Requires(arg > 0)
- Throw New NotSupportedException
- End Function
- End Class
La classe InterfacciaContract funge da contenitore per i contratti dei metodi esposti dalla Interfaccia.
Il codice presente nei metodi dell’interfaccia implementati dalla classe contenitore non viene eseguito se non quello relativo alla definizione dei contratti.
Il problema nasce quando abbiamo una interfaccia generica (ovvero il cui comportamento dipende da uno o più tipi). In questo caso come definiamo la classe “contenitore” dei contratti e come indichiamo il tipo dell’interfaccia nell’attributo ContractClassFor?
In questo caso si può utilizzare la sintassi NomeTipo(Of ) come mostrato nel seguente esempio:
- <ContractClass(GetType(RepositoryContract(Of )))>
- Public Interface IRepository(Of T)
- Function GetAll() As IEnumerable(Of T)
- Function GetSingle(id As Integer) As IEnumerable(Of T)
- End Interface
- <ContractClassFor(GetType(IRepository(Of )))>
- Public MustInherit Class RepositoryContract(Of T)
- Implements IRepository(Of T)
- Public Function GetSingle(id As Integer) As System.Collections.Generic.IEnumerable(Of T) Implements IRepository(Of T).GetSingle
- Contract.Requires(id > 0)
- Throw New NotImplementedException()
- End Function
- Public Function GetAll() As System.Collections.Generic.IEnumerable(Of T) Implements IRepository(Of T).GetAll
- Throw New NotImplementedException()
- End Function
- End Class
Nel momento in cui definiamo una classe reale imponendo il tipo T, avremo che il contratto viene “ereditato” automaticamente:
- Public Class Repository
- Implements IRepository(Of Entity)
- Public Function GetSingle(id As Integer) As System.Collections.Generic.IEnumerable(Of Entity) Implements IRepository(Of Entity).GetSingle
- Return Nothing
- End Function
- Public Function GetAll() As System.Collections.Generic.IEnumerable(Of Entity) Implements IRepository(Of Entity).GetAll
- Return Nothing
- End Function
- End Class
senza che noi si debba riscriverlo.
Nel caso in cui l’interfaccia dipenda da più tipi è sufficiente scrivere le opportune virgole:
- <ContractClass(GetType(RepositoryContract(Of ,)))>
- Public Interface IRepository(Of T1, T2)
- Function GetAll() As IEnumerable(Of T1)
- Function GetSingle(id As Integer) As IEnumerable(Of T1)
- End Interface
- <ContractClassFor(GetType(IRepository(Of ,)))>
- Public MustInherit Class RepositoryContract(Of T1, T2)
- Implements IRepository(Of T1, T2)
- Public Function GetAll() As System.Collections.Generic.IEnumerable(Of T1) Implements IRepository(Of T1, T2).GetAll
- Throw New NotImplementedException()
- End Function
- Public Function GetSingle(id As Integer) As System.Collections.Generic.IEnumerable(Of T1) Implements IRepository(Of T1, T2).GetSingle
- Contract.Requires(id > 0)
- Throw New NotImplementedException()
- End Function
- End Class
Commenti