Varians i ombud (Visual Basic)
.NET Framework 3.5 introducerade variansstöd för matchande metodsignaturer med ombudstyper i alla ombud i C# och Visual Basic. Det innebär att du kan tilldela ombud inte bara metoder som har matchande signaturer, utan även metoder som returnerar fler härledda typer (kovarians) eller som accepterar parametrar som har mindre härledda typer (kontravarians) än de som anges av ombudstypen. Detta omfattar både allmänna och icke-generiska ombud.
Tänk till exempel på följande kod, som har två klasser och två ombud: generiska och icke-generiska.
Public Class First
End Class
Public Class Second
Inherits First
End Class
Public Delegate Function SampleDelegate(ByVal a As Second) As First
Public Delegate Function SampleGenericDelegate(Of A, R)(ByVal a As A) As R
När du skapar ombud av typerna SampleDelegate
eller SampleDelegate(Of A, R)
kan du tilldela någon av följande metoder till dessa ombud.
' Matching signature.
Public Shared Function ASecondRFirst(
ByVal second As Second) As First
Return New First()
End Function
' The return type is more derived.
Public Shared Function ASecondRSecond(
ByVal second As Second) As Second
Return New Second()
End Function
' The argument type is less derived.
Public Shared Function AFirstRFirst(
ByVal first As First) As First
Return New First()
End Function
' The return type is more derived
' and the argument type is less derived.
Public Shared Function AFirstRSecond(
ByVal first As First) As Second
Return New Second()
End Function
Följande kodexempel illustrerar den implicita konverteringen mellan metodsignaturen och ombudstypen.
' Assigning a method with a matching signature
' to a non-generic delegate. No conversion is necessary.
Dim dNonGeneric As SampleDelegate = AddressOf ASecondRFirst
' Assigning a method with a more derived return type
' and less derived argument type to a non-generic delegate.
' The implicit conversion is used.
Dim dNonGenericConversion As SampleDelegate = AddressOf AFirstRSecond
' Assigning a method with a matching signature to a generic delegate.
' No conversion is necessary.
Dim dGeneric As SampleGenericDelegate(Of Second, First) = AddressOf ASecondRFirst
' Assigning a method with a more derived return type
' and less derived argument type to a generic delegate.
' The implicit conversion is used.
Dim dGenericConversion As SampleGenericDelegate(Of Second, First) = AddressOf AFirstRSecond
Fler exempel finns i Använda varians i ombud (Visual Basic) och Using Variance for Func and Action Generic Delegates (Visual Basic).
Varians i parametrar av allmän typ
I .NET Framework 4 och senare kan du aktivera implicit konvertering mellan ombud, så att allmänna ombud som har olika typer som anges av generiska typparametrar kan tilldelas till varandra, om typerna ärvs från varandra enligt variansen.
Om du vill aktivera implicit konvertering måste du uttryckligen deklarera generiska parametrar i ett ombud som covariant eller kontravariant med hjälp av nyckelordet in
eller out
.
Följande kodexempel visar hur du kan skapa ett ombud som har en covariant allmän typparameter.
' Type T is declared covariant by using the out keyword.
Public Delegate Function SampleGenericDelegate(Of Out T)() As T
Sub Test()
Dim dString As SampleGenericDelegate(Of String) = Function() " "
' You can assign delegates to each other,
' because the type T is declared covariant.
Dim dObject As SampleGenericDelegate(Of Object) = dString
End Sub
Om du endast använder avvikelsestöd för att matcha metodsignaturer med ombudstyper och inte använder nyckelorden in
och out
kanske du upptäcker att du ibland kan instansiera ombud med identiska lambda-uttryck eller metoder, men du kan inte tilldela en delegat till en annan.
I följande kodexempel SampleGenericDelegate(Of String)
kan inte uttryckligen konverteras till SampleGenericDelegate(Of Object)
, men String
ärver Object
. Du kan åtgärda det här problemet genom att markera den generiska parametern T
med nyckelordet out
.
Public Delegate Function SampleGenericDelegate(Of T)() As T
Sub Test()
Dim dString As SampleGenericDelegate(Of String) = Function() " "
' You can assign the dObject delegate
' to the same lambda expression as dString delegate
' because of the variance support for
' matching method signatures with delegate types.
Dim dObject As SampleGenericDelegate(Of Object) = Function() " "
' The following statement generates a compiler error
' because the generic type T is not marked as covariant.
' Dim dObject As SampleGenericDelegate(Of Object) = dString
End Sub
Allmänna ombud som har parametrar av varianttyp i .NET Framework
.NET Framework 4 introducerade variansstöd för generiska typparametrar i flera befintliga generiska ombud:
Action
ombud från System namnområdet, till exempel, Action<T> och Action<T1,T2>Func
ombud från System namnområdet, till exempel, Func<TResult> och Func<T,TResult>Ombudet Predicate<T>
Ombudet Comparison<T>
Ombudet Converter<TInput,TOutput>
Mer information och exempel finns i Using Variance for Func and Action Generic Delegates (Visual Basic).
Deklarera parametrar av varianttyp i allmänna ombud
Om ett allmänt ombud har parametrar av typen covariant eller kontravariant allmän typ kan det kallas för ett generiskt variantdelegat.
Du kan deklarera en generisk typparameter covariant i ett allmänt ombud med hjälp av nyckelordet out
. Den covarianta typen kan endast användas som en metodreturtyp och inte som en typ av metodargument. I följande kodexempel visas hur du deklarerar ett samvariant generiskt ombud.
Public Delegate Function DCovariant(Of Out R)() As R
Du kan deklarera en generisk typparameter kontravariant i ett allmänt ombud med hjälp av nyckelordet in
. Typen contravariant kan endast användas som en typ av metodargument och inte som en metodreturtyp. I följande kodexempel visas hur du deklarerar ett kontravariant generiskt ombud.
Public Delegate Sub DContravariant(Of In A)(ByVal a As A)
Viktigt!
ByRef
parametrar i Visual Basic kan inte markeras som variant.
Det är också möjligt att stödja både varians och kovarians i samma ombud, men för olika typparametrar. Detta visas i följande exempel.
Public Delegate Function DVariant(Of In A, Out R)(ByVal a As A) As R
Instansiera och anropa generiska variantdelegater
Du kan instansiera och anropa variantdelegater precis som du instansierar och anropar invarianta ombud. I följande exempel instansieras ombudet av ett lambda-uttryck.
Dim dvariant As DVariant(Of String, String) = Function(str) str + " "
dvariant("test")
Kombinera generiska variantdelegater
Du bör inte kombinera variantdelegater. Metoden Combine stöder inte variantdelegatkonvertering och förväntar sig att ombuden ska vara av exakt samma typ. Detta kan leda till ett körningsfel när du kombinerar ombud antingen med hjälp Combine av metoden (i C# och Visual Basic) eller med hjälp av operatorn +
(i C#), som du ser i följande kodexempel.
Dim actObj As Action(Of Object) = Sub(x) Console.WriteLine("object: {0}", x)
Dim actStr As Action(Of String) = Sub(x) Console.WriteLine("string: {0}", x)
' The following statement throws an exception at run time.
' Dim actCombine = [Delegate].Combine(actStr, actObj)
Varians i generiska typparametrar för värde- och referenstyper
Avvikelse för generiska typparametrar stöds endast för referenstyper. Det går till exempel DVariant(Of Int)
inte att implicit konvertera till DVariant(Of Object)
eller DVariant(Of Long)
, eftersom heltal är en värdetyp.
I följande exempel visas att variansen i generiska typparametrar inte stöds för värdetyper.
' The type T is covariant.
Public Delegate Function DVariant(Of Out T)() As T
' The type T is invariant.
Public Delegate Function DInvariant(Of T)() As T
Sub Test()
Dim i As Integer = 0
Dim dInt As DInvariant(Of Integer) = Function() i
Dim dVariantInt As DVariant(Of Integer) = Function() i
' All of the following statements generate a compiler error
' because type variance in generic parameters is not supported
' for value types, even if generic type parameters are declared variant.
' Dim dObject As DInvariant(Of Object) = dInt
' Dim dLong As DInvariant(Of Long) = dInt
' Dim dVariantObject As DInvariant(Of Object) = dInt
' Dim dVariantLong As DInvariant(Of Long) = dInt
End Sub
Avslappnad delegatkonvertering i Visual Basic
Avslappnad delegatkonvertering ger större flexibilitet när det gäller att matcha metodsignaturer med ombudstyper. Du kan till exempel utelämna parameterspecifikationer och utelämna funktionsreturvärden när du tilldelar en metod till ett ombud. Mer information finns i Avslappnad delegatkonvertering.