|
Dit artikel is gepubliceerd op zondag 31 juli 2011 op vbvoorbeelden, bezoek de website voor een recente versie van dit artikel of andere artikels.
11.4.1. Geen Contravariant Parametertype ToegestaanBij het herdefiniëren van Method1 in DerivedClass (1) zou je gezien Liskovs Substitution Principle ( LSP ) verwachten dat je het argumenttype het basistype zou mogen zijn van Apple, namelijk Fruit. Dit is wat men contravariance noemt, of op zijn minst een contravariant parametertype ( basistype als parametertype ). Visual Basic 2010 Broncode Class Fruit Public Function MethodA() As String End FunctionEnd ClassClass Apple : Inherits Fruit Public Function MethodB() As String End FunctionEnd ClassClass GrannySmith : Inherits Apple Public Function MethodC() As String End FunctionEnd ClassClass JonaGold : Inherits Apple Public Function MethodD() As String End FunctionEnd ClassClass BaseClass Public Overridable Sub Method1( ByVal argument As Apple) End SubEnd ClassClass DerivedClass : Inherits BaseClass Public Overrides Sub Method1( ByVal argument As Apple) End Sub Public Overrides Sub Method1( ByVal argument As Fruit) Console.WriteLine(argument.MethodA()) End Sub End ClassClass ExpectedContraVariance Public Shared Sub Main() Dim grannySmith1 As GrannySmith = New GrannySmith Dim jonaGold1 As JonaGold = New JonaGold Dim object1 As BaseClass = New DerivedClass object1.Method1(grannySmith1) End SubEnd ClassDownload Visual Basic 2010 Broncode Download Visual C# Sourcecode
Bij de call naar Method1 op object1 (2) kunnen we enkel een Apple, en dus geen Fruit doorgeven. De signatuur van Method1 in BaseClass (3) definieert immers een argument van type Apple te verwachten. De implementatie van Method1 in DerivedClass (4) kan, gezien zijn signatuur (1), enkel werken op een variable van statisch type Fruit. Het dynamisch type van object1 mag dan nog wel DerivedClass zijn, het wordt gebonden aan een implementatie (4) die er niets meer mee zal doen dan wat met een Fruit kan doen, bijvoorbeeld aanroepen van MethodA, wat ook met een Apple kan.
Dit mag dan al te verwachten zijn, toch is dit niet mogelijk. Bij redefinition mag men het parametertype niet veranderen (6), dit is wat men ook wel invariance noemt.
Dat een covariant parametertype ( een afgeleid parametertype ) niet mogelijk is, is gezien LSP logisch. De call naar Method1 op object1 verwacht een argument van type Apple, het statisch type van object1 is immers BaseClass, dat Method1 heeft gedefinieert als een procedure die een Apple verwacht. Als we in de herdefinitie (7) opeens een subtype als parametertype nemen, dan loopt de call op regel (8) fout. De doorgegeven JonaGold is immers geen GrannySmith. boven
11.4.2. Geen Covariant Returntype ToegestaanOok het returntype mag bij een herdefinitie niet veranderen. Toch zou men op zijn minst de mogelijkheid verwachten, gezien LSP, om een covariant returntype te kunnen kiezen. Op de expressie van type Apple die de object1.Method2() call vormt, kan je MethodB aanroepen. Dat de call naar MethodB wordt gebonden aan de implementatie van een GrannySmith, die Method2 van DerivedClass oplevert, is dan ook geen enkel probleem.
Toch is dit niet toegestaan. Het returntype mag in een herdefinitie niet veranderen.
Dat men in de herdefinitie geen basistype als returntype mag gebruiken is gezien LSP logisch. Stel dat Method2 een Fruit object oplevert, zoals implementatie (1), dan zou men MethodB die daarop wordt aangeroepen, niet kunnen binden aan een implementatie. Een Fruit heeft immers geen MethodB. Visual Basic 2010 Broncode Partial Class BaseClass Public Overridable Function Method2() As Apple Method2 = New Apple End FunctionEnd ClassPartial Class DerivedClass : Inherits BaseClass Public Overrides Function Method2() As Apple Method2 = New Apple End Function Public Overrides Function Method2() As GrannySmith Method2 = New GrannySmith End Function End ClassClass ExpectedCoVariance Public Shared Sub Main() Dim grannySmith1 As GrannySmith = New GrannySmith Dim jonaGold1 As JonaGold = New JonaGold Dim object1 As BaseClass = New DerivedClass Console.WriteLine(object1.Method2().MethodB()) End SubEnd ClassDownload Visual Basic 2010 Broncode Download Visual C# Sourcecode
Dit artikel is gepubliceerd op zondag 31 juli 2011 op vbvoorbeelden, bezoek de website voor een recente versie van dit artikel of andere artikels.
|