实现具有i等于t的ienumerable的实现. [英] implement ienumerable with ienumerable<T>

查看:65
本文介绍了实现具有i等于t的ienumerable的实现.的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

在通用集合类中用IEnumerable(通用)接口实现IEnumerable(非通用)有什么需要?
有关msdn状态的代码示例(链接- http://msdn.microsoft.com/zh-CN/library/9eekhta0(v = vs.110).aspx )

What is the need to implement IEnumerable (Non-generic) with IEnumerable (Generic) interface in a generic collection class?
A code example on msdn states (Link - http://msdn.microsoft.com/en-us/library/9eekhta0(v=vs.110).aspx)

public class App
{
// Excercise the Iterator and show that it's more 
// performant. 
public static void Main()
{
    TestStreamReaderEnumerable();
    TestReadingFile();
}

public static void TestStreamReaderEnumerable()
{
    // Check the memory before the iterator is used. 
    long memoryBefore = GC.GetTotalMemory(true);

    // Open a file with the StreamReaderEnumerable and check for a string. 
    var stringsFound =
        from line in new StreamReaderEnumerable(@"c:\\temp\\tempFile.txt")
        where line.Contains("string to search for")
        select line;


    Console.WriteLine("Found: " + stringsFound.Count());

    // Check the memory after the iterator and output it to the console. 
    long memoryAfter = GC.GetTotalMemory(false);
    Console.WriteLine("Memory Used With Iterator = \t"
        + string.Format(((memoryAfter - memoryBefore) / 1000).ToString(), "n") + "kb");
}

public static void TestReadingFile()
{
    long memoryBefore = GC.GetTotalMemory(true);
    StreamReader sr = File.OpenText("c:\\temp\\tempFile.txt");

    // Add the file contents to a generic list of strings.
    List<string> fileContents = new List<string>();
    while (!sr.EndOfStream) {
        fileContents.Add(sr.ReadLine());
    }

    // Check for the string. 
    var stringsFound = 
        from line in fileContents
        where line.Contains("string to search for")
        select line;

    sr.Close();
    Console.WriteLine("Found: " + stringsFound.Count());

    // Check the memory after when the iterator is not used, and output it to the console. 
    long memoryAfter = GC.GetTotalMemory(false);
    Console.WriteLine("Memory Used Without Iterator = \t" + 
        string.Format(((memoryAfter - memoryBefore) / 1000).ToString(), "n") + "kb");

}
}

// A custom class that implements IEnumerable(T). When you implement IEnumerable(T),  
// you must also implement IEnumerable and IEnumerator(T). 
public class StreamReaderEnumerable : IEnumerable<string>
{

private string _filePath;
public StreamReaderEnumerable(string filePath)
{
    _filePath = filePath;
}


// Must implement GetEnumerator, which returns a new StreamReaderEnumerator. 
public IEnumerator<string> GetEnumerator()
{
    return new StreamReaderEnumerator(_filePath);
}

// Must also implement IEnumerable.GetEnumerator, but implement as a private method. 
private IEnumerator GetEnumerator1()
{
    return this.GetEnumerator();
}
IEnumerator IEnumerable.GetEnumerator()
{
    return GetEnumerator1();
}

} 

// When you implement IEnumerable(T), you must also implement IEnumerator(T),  
// which will walk through the contents of the file one line at a time. 
// Implementing IEnumerator(T) requires that you implement IEnumerator and IDisposable. 
public class StreamReaderEnumerator : IEnumerator<string>
{
private StreamReader _sr;
public StreamReaderEnumerator(string filePath)
{
    _sr = new StreamReader(filePath);
}

private string _current;
// Implement the IEnumerator(T).Current publicly, but implement  
// IEnumerator.Current, which is also required, privately. 
public string Current
{

    get
    {
        if (_sr == null || _current == null)
        {
            throw new InvalidOperationException();
        }

        return _current;
    }
}

private object Current1
{

    get { return this.Current; }
}
object IEnumerator.Current
{
    get { return Current1; }
}

// Implement MoveNext and Reset, which are required by IEnumerator. 
public bool MoveNext()
{

    _current = _sr.ReadLine();
    if (_current == null)
        return false;
    return true;
}

public void Reset()
{
    _sr.DiscardBufferedData();
    _sr.BaseStream.Seek(0, SeekOrigin.Begin);
    _current = null;
}
// Implement IDisposable, which is also implemented by IEnumerator(T). 
private bool disposedValue = false;
public void Dispose()
{
    Dispose(true);
    GC.SuppressFinalize(this);
}
protected virtual void Dispose(bool disposing)
{
    if (!this.disposedValue)
    {
        if (disposing)
        {
            // Dispose of managed resources.
        }
        _current = null;
        _sr.Close();
        _sr.Dispose();
    }

    this.disposedValue = true;
}

 ~StreamReaderEnumerator()
{
    Dispose(false);
}
// This example displays output similar to the following: 
//Found: 2 
//Memory Used With Iterator =     33kb 
//Found: 2 
//Memory Used Without Iterator =  206kb
}

推荐答案

需求仅仅是因为接口的实现方式.在.NET 1.1中,没有泛型,因此IEnumerable没有泛型支持,只公开了object.这引入了装箱(这也是为什么编译器还支持foreach的基于模式的实现,而与IEnumerable无关的原因)的原因.在C#2中,我们得到了IEnumerable<T>.将所有类型化迭代都视为未类型化也是很有用的,因此:

The need is simply because of how the interfaces are implemented. In .NET 1.1, there were no generics, so IEnumerable has no generic support and only exposes object. This introduces boxing (and is why the compiler also supports a pattern-based implementation for foreach, independent of IEnumerable). In C# 2, we got IEnumerable<T>. It is useful to consider all typed iterations as also being untyped, hence:

IEnumerable<T> : IEnumerable

其中,IEnumerable<T>与通用类型一样重新声明GetEnumerator.但是,由于这些现在是具有不同签名的不同方法,因此需要2种不同的实现.

where IEnumerable<T> re-declares GetEnumerator as with the generic type. However, since these are now different methods with different signatures, it needs 2 different implementations.

这篇关于实现具有i等于t的ienumerable的实现.的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆