我在编译时遇到以下错误消息:
“无效的方差:类型参数'T'必须在'ConsoleApplication1.IRepository.GetAll()'上无效。”T'是协变的。“
以下是我的代码:
class Program
{
static void Main(string[] args)
{
IRepository<BaseClass> repository;
repository = new RepositoryDerived1<Derived1>();
Console.ReadLine();
}
}
public abstract class BaseClass
{
}
public class Derived1 : BaseClass
{
}
public interface IRepository<out T> where T: BaseClass, new()
{
IList<T> GetAll();
}
public class Derived2 : BaseClass
{
}
public abstract class RepositoryBase<T> : IRepository<T> where T: BaseClass, new()
{
public abstract IList<T> GetAll();
}
public class RepositoryDerived1<T> : RepositoryBase<T> where T: BaseClass, new()
{
public override IList<T> GetAll()
{
throw new NotImplementedException();
}
}
我需要的是能够像这样使用上述类:
IRepository存储库;
或
RepositoryBase存储库;
然后我希望能够分配这样的东西:
repository = new RepositoryDerived1();
但是它在IRepository类上给出了编译时错误。
如果我从IRepository类中删除“out”关键字,它会给我另一个错误
“RepositoryDerived1”无法转换为“IRepository”。
为什么以及如何解决它?
由于
答案 0 :(得分:4)
IList<T>
不是协变的。如果您将IList<T>
更改为IEnumerable<T>
,并从: new()
中删除IRepository<out T>
约束(因为抽象基类不满足该约束),它将起作用:
public interface IRepository<out T> where T : BaseClass
{
IEnumerable<T> GetAll();
}
public abstract class RepositoryBase<T> : IRepository<T> where T : BaseClass, new()
{
public abstract IEnumerable<T> GetAll();
}
public class RepositoryDerived1<T> : RepositoryBase<T> where T : BaseClass, new()
{
public override IEnumerable<T> GetAll()
{
throw new NotImplementedException();
}
}