J'ai trouvé un motif qui semble plutôt bien. Il est inspiré par quelqu'un posté sur CodeProject.com - en utilisant une liste pour garder une trace des consommables; raiiBase (de T) est une classe de base adaptée à toute classe dont le constructeur prend un seul paramètre. Le constructeur de la classe doit être protégé et la construction doit être effectuée par la méthode usine. Le constructeur statique makeRaii() prend un délégué à une fonction d'usine, qui doit accepter une pile (de iDisposable) avec un paramètre du type attendu de la classe.Une utilisation de l'échantillon:
Public Class RaiiTest
Inherits raiiBase(Of String)
Dim thing1 As testDisposable = RAII(New testDisposable("Moe " & creationParam, "a"))
Dim thing2 As testDisposable = RAII(New testDisposable("Larry " & creationParam, "b"))
Dim thing3 As testDisposable = RAII(New testDisposable("Shemp " & creationParam, "c"))
Dim thing4 As testDisposable = RAII(New testDisposable("Curly " & creationParam, "d"))
Protected Sub New(ByVal dispList As Stack(Of IDisposable), ByVal newName As String)
MyBase.New(dispList, newName)
End Sub
Private Shared Function _newRaiiTest(ByVal dispList As Stack(Of IDisposable), ByVal theName As String) As RaiiTest
Return New RaiiTest(dispList, theName)
End Function
Public Shared Function newRaiiTest(ByVal theName As String) As RaiiTest
Return makeRaii(Of RaiiTest)(AddressOf _newRaiiTest, theName)
End Function
Shared Sub test(ByVal st As String)
Try
Using it As RaiiTest = newRaiiTest(st)
Debug.Print("Now using object")
End Using
Debug.Print("No exceptions thrown")
Catch ex As raiiException
Debug.Print("Output exception: " & ex.Message)
If ex.InnerException IsNot Nothing Then Debug.Print("Inner exception: " & ex.InnerException.Message)
For Each exx As Exception In ex.DisposalExceptions
Debug.Print("Disposal exception: " & exx.Message)
Next
Catch ex As Exception
Debug.Print("Misc. exception: " & ex.Message)
End Try
End Sub
End Class
Depuis raiiTest raiiBase hérite (String), pour créer une instance de classe, appelez newRaiiTest avec un paramètre de chaîne. RAII() est une fonction générique qui va enregistrer son argument en tant que iDisposable qui aura besoin d'être nettoyé, puis le retourner. Tous les objets jetables enregistrés seront éliminés lorsque Dispose est appelé sur l'objet principal ou lorsqu'une exception est générée dans la construction de l'objet principal.
est ici la classe riaaBase:
Option Strict On
Class raiiException
Inherits Exception
ReadOnly _DisposalExceptions() As Exception
Sub New(ByVal message As String, ByVal InnerException As Exception, ByVal allInnerExceptions As Exception())
MyBase.New(message, InnerException)
_DisposalExceptions = allInnerExceptions
End Sub
Public Overridable ReadOnly Property DisposalExceptions() As Exception()
Get
Return _DisposalExceptions
End Get
End Property
End Class
Public Class raiiBase(Of T)
Implements IDisposable
Protected raiiList As Stack(Of IDisposable)
Protected creationParam As T
Delegate Function raiiFactory(Of TT As raiiBase(Of T))(ByVal theList As Stack(Of IDisposable), ByVal theParam As T) As TT
Shared Function CopyFirstParamToSecondAndReturnFalse(Of TT)(ByVal P1 As TT, ByRef P2 As TT) As Boolean
P2 = P1
Return False
End Function
Shared Function makeRaii(Of TT As raiiBase(Of T))(ByVal theFactory As raiiFactory(Of TT), ByVal theParam As T) As TT
Dim dispList As New Stack(Of IDisposable)
Dim constructionFailureException As Exception = Nothing
Try
Return theFactory(dispList, theParam)
Catch ex As Exception When CopyFirstParamToSecondAndReturnFalse(ex, constructionFailureException)
' The above statement let us find out what exception occurred without having to catch and rethrow
Throw ' Should never happen, since we should have returned false above
Finally
If constructionFailureException IsNot Nothing Then
zapList(dispList, constructionFailureException)
End If
End Try
End Function
Protected Sub New(ByVal DispList As Stack(Of IDisposable), ByVal Params As T)
Me.raiiList = DispList
Me.creationParam = Params
End Sub
Public Shared Sub zapList(ByVal dispList As IEnumerable(Of IDisposable), ByVal triggerEx As Exception)
Using theEnum As IEnumerator(Of IDisposable) = dispList.GetEnumerator
Try
While theEnum.MoveNext
theEnum.Current.Dispose()
End While
Catch ex As Exception
Dim exList As New List(Of Exception)
exList.Add(ex)
While theEnum.MoveNext
Try
theEnum.Current.Dispose()
Catch ex2 As Exception
exList.Add(ex2)
End Try
End While
Throw New raiiException("RAII failure", triggerEx, exList.ToArray)
End Try
End Using
End Sub
Function RAII(Of U As IDisposable)(ByVal Thing As U) As U
raiiList.Push(Thing)
Return Thing
End Function
Shared Sub zap(ByVal Thing As IDisposable)
If Thing IsNot Nothing Then Thing.Dispose()
End Sub
Private raiiBaseDisposeFlag As Integer = 0 ' To detect redundant calls
' IDisposable
Protected Overridable Sub Dispose(ByVal disposing As Boolean)
If disposing AndAlso Threading.Interlocked.Exchange(raiiBaseDisposeFlag, 1) = 0 Then
zapList(raiiList, Nothing)
End If
End Sub
#Region " IDisposable Support "
' This code added by Visual Basic to correctly implement the disposable pattern.
Public Sub Dispose() Implements IDisposable.Dispose
' Do not change this code. Put cleanup code in Dispose(ByVal disposing As Boolean) above.
Dispose(True)
GC.SuppressFinalize(Me)
End Sub
#End Region
End Class
Notez qu'un type d'exception personnalisée sera levée si l'élimination échoue pour tout ou partie des objets à usage unique enregistrés. InnerException indiquera si le constructeur a échoué; pour voir quel (s) broyeur (s) a échoué, vérifiez DisposalExceptions.
-1 Cela ne fonctionne pas si le constructeur lève une exception. Le nouvel objet n'est jamais retourné, donc il n'y a rien à appeler Dispose. – chilltemp
Hmmmm ... éclaté. –