Почему вы не можете использовать небезопасное ключевое слово в контексте итератора? - PullRequest
6 голосов
/ 11 февраля 2010

Рассматривая этот вопрос, на который Джон отлично справился, ответив ... ' Как реверсивно читать текстовый файл с помощью итератора '. И был похожий вопрос, на который я ответил, используя указатели hocus pocus .. ' .net есть способ прочитать текстовый файл снизу вверх ', прежде чем он закроется ....

Теперь я решил попытаться решить эту проблему с помощью указателей, ладно, по краям это выглядит грубо и грубо ...

public class ReadChar : IEnumerable<char>
{
    private Stream _strm = null;
    private string _str = string.Empty;
    public ReadChar(string s)
    {
        this._str = s;
    }
    public ReadChar(Stream strm)
    {
        this._strm = strm;
    }
    public IEnumerator<char> GetEnumerator()
    {
        if (this._strm != null && this._strm.CanRead && this._strm.CanSeek)
        {
            return ReverseReadStream();
        }
        if (this._str.Length > 0)
        {
            return ReverseRead();
        }
        return null;
    }

    private IEnumerator<char> ReverseReadStream()
    {
        long lIndex = this._strm.Length;
        while (lIndex != 0 && this._strm.Position != 0)
        {
            this._strm.Seek(lIndex--, SeekOrigin.End);
            int nByte = this._strm.ReadByte();
            yield return (char)nByte; 
        }
    }

    private IEnumerator<char> ReverseRead()
    {
        unsafe
        {
            fixed (char* beg = this._str)
            {
                char* p = beg + this._str.Length;
                while (p-- != beg)
                {
                    yield return *p;
                }
            }
        }
    }
    IEnumerator IEnumerable.GetEnumerator()
    {
        return GetEnumerator();
    }
}

но обнаружил, что компилятор C # не может справиться с этим с помощью этой реализации, но был опустошен, когда компилятор C # отказался с ошибкой CS1629 - «Небезопасный код может не отображаться в итераторах»

Почему это так?

Ответы [ 3 ]

6 голосов
/ 11 февраля 2010

Эрик Липперт имеет отличную запись в блоге на эту тему здесь: Блоки итераторов, часть шестая: почему нет небезопасного кода?

6 голосов
/ 12 февраля 2010

Что я хочу знать, так это то, почему вы вообще используете указатели для этого. Почему бы просто не сказать:

private IEnumerator<char> ReverseRead()
{
    int len = _str.Length;
    for(int i = 0; i < len; ++i)
        yield return _str[len - i - 1];
}

В чем убедительная польза от работы с указателями?

1 голос
/ 11 февраля 2010

Это просто часть спецификации C #:

26.1 Блоки итераторов ... Ошибка времени компиляции для блока итератора, содержащего небезопасный контекст (§27.1). Блок итератора всегда определяет безопасный контекст, даже если его объявление вложено в небезопасный контекст.

Предположительно, код, который генерируется компилятором, должен быть проверяемым, чтобы не иметь ярлыка «небезопасный». Если вы хотите использовать указатели, вам придется реализовать IEnumerator самостоятельно.

...