将列表泛型getenumator强制转换为接口失败



GetEnumerator强制转换到接口失败
没有编译错误
消息索引为无穷大的运行时失败
如果我直接使用结构体Word1252而不使用接口,它可以工作

namespace WordEnumerable
{
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
            WordEnum wordEnum = new WordEnum();
            Debug.WriteLine(wordEnum[0].GetType().ToString());
            Debug.WriteLine(wordEnum[0].Value);
            Debug.WriteLine(wordEnum.Count.ToString());
            foreach (iWord w in wordEnum)  // fails here
            {
            }
        }
    }
}
public interface iWord
{
    Int32 Key { get; }
    String Value { get; }
}
public class WordEnum : IEnumerable<iWord>
{
    private static List<Word1252> words1252 = new List<Word1252>();
    IEnumerator<iWord> IEnumerable<iWord>.GetEnumerator()
    {
        return ((IEnumerable<iWord>)words1252).GetEnumerator();
    }
    public struct Word1252 : iWord
    {
        public UInt64 packed;
        public Int32 Key   { get { return (Int32)((packed >> 27) & ((1 << 25) - 1)); } }
        public Byte Length { get { return (Byte) ((packed >> 59) & ((1 <<  5) - 1)); } }
        public String Value { get { return Key.ToString(); } }
        public Word1252(UInt64 Packed) { packed = Packed; }
    }

简而言之就是upcastingIEnumerable<Word1252>IEnumerable<IWord>

这需要covariance工作。

即使IEnumerable被标记为out(协变)
协方差不适用于'值类型' -即您拥有的结构体接口定义。

。看到……

这是c# 4中的协方差错误吗?
(或者有点不同,但归结为相同的问题)
为什么IEnumerable转换为IEnumerable?

你可以像

那样定义你的列表
private static List<iWord> words1252 = new List<iWord>();  

或者像这样定义你的枚举数:

IEnumerator<iWord> IEnumerable<iWord>.GetEnumerator()
{
    foreach (var word in words1252)
        yield return word;
}

最新更新