The following code:
public interface ISomeData
{
IEnumerable<string> Data { get; }
}
public class MyData : ISomeData
{
private List<string> m_MyData = new List<string>();
public List<string> Data { get { return m_MyData; } }
}
Produces the following error:
error CS0738: ‘InheritanceTest.MyData’
does not implement interface member
‘InheritanceTest.ISomeData.Data’.
‘InheritanceTest.MyData.Data’ cannot
implement
‘InheritanceTest.ISomeData.Data’
because it does not have the matching
return type of
‘System.Collections.Generic.IEnumerable’.
Since a List<T> implements IEnumerable<T>, one would think that my class would implement the interface. Can someone explain what the rationale is for this not compiling?
As I can see it, there are two possible solutions:
- Change the interface to be more specific and require IList be implemented.
- Change my class (MyData) to return IEnumerable and implement the original interface.
Now suppose I also have the following code:
public class ConsumerA
{
static void IterateOverCollection(ISomeData data)
{
foreach (string prop in data.MyData)
{
/*do stuff*/
}
}
}
public class ConsumerB
{
static void RandomAccess(MyData data)
{
data.Data[1] = "this line is invalid if MyPropList return an IEnumerable<string>";
}
}
I could change my interface to require IList to be implemented (option 1), but that limits who can implement the interface and the number of classes that can be passed into ConsumerA. Or, I could change implementation (class MyData) so that it returns an IEnumerable instead of a List (option 2), but then ConsumerB would have to be rewritten.
This seems to be a shortcoming of C# unless someone can enlighten me.
Unfortunately, the return type must match. What you are looking for is called ‘return type covariance’ and C# doesn’t support that.
http://connect.microsoft.com/VisualStudio/feedback/ViewFeedback.aspx?FeedbackID=90909
Eric Lippert, senior developer on C# Compiler team, mentions on his blog that they don’t plan to support return type covariance.
http://blogs.msdn.com/ericlippert/archive/2008/05/07/covariance-and-contravariance-part-twelve-to-infinity-but-not-beyond.aspx
It’s worth reading Eric’s articles on covariance and contravariance.
http://blogs.msdn.com/ericlippert/archive/tags/Covariance+and+Contravariance/default.aspx