Является ли повышение производительности «реальной» целью существования заявления «GOTO»? - PullRequest
2 голосов
/ 29 января 2011

Сегодня я открыл одну из реализаций регулярных выражений Microsoft .Net, и меня поразило, что это может быть единственной причиной существования оператора goto в языке, подобном C # (в котором основной упор делается на «ясность дляконцепция разработчика, где goto, кажется, вообще не имеет места) и, возможно, также единственная причина, по которой Microsoft использует его в реализации библиотек CLR - для повышения производительности.Я также помню подобную оптимизацию при рендеринге страниц / элементов управления Microsoft ASP.NET в System.Web.UI.dll.Будет ли это правильным (или документально подтвержденным?) Предположением?Вы видели, как Microsoft использовала при любых других обстоятельствах, кроме как для улучшения времени выполнения кода?Спасибо.

Ниже приведен отрывок ( НЕ СЛЕДУЕТ РАССМОТРЕТЬ В ДЕТАЛЯХ, ПРОСТО СКАНИРОВАТЬ ЕГО ) из собственной реализации регулярных выражений Microsoft, где они снова широко используют goto, что, по-видимому, является лишь повышением производительности:

public override void Go() 
{
    int num4;
    int num5;
    string runtext = base.runtext;
    int runtextstart = base.runtextstart;
    int runtextbeg = base.runtextbeg;
    int runtextend = base.runtextend;
    int runtextpos = base.runtextpos;
    int[] runtrack = base.runtrack;
    int runtrackpos = base.runtrackpos;
    int[] runstack = base.runstack;
    int runstackpos = base.runstackpos;
    runtrack[--runtrackpos] = runtextpos;
    runtrack[--runtrackpos] = 0;
    runstack[--runstackpos] = runtextpos;
    runtrack[--runtrackpos] = 1;
    if ((((runtextpos != base.runtextstart) || 
       (4 > (runtextend - runtextpos))) || 
       ((runtext[runtextpos] != '<') || 
       (runtext[runtextpos + 1] != '%'))) || 
       ((runtext[runtextpos + 2] != '-') || 
       (runtext[runtextpos + 3] != '-')))
    {
        goto Label_02F8;
    }
    runtextpos += 4;
    runstack[--runstackpos] = -1;
    runtrack[--runtrackpos] = 1;
    goto Label_0213;
Label_0161:
    if (num5 > num4)
    {
        runtrack[--runtrackpos] = (num5 - num4) - 1;
        runtrack[--runtrackpos] = runtextpos - 1;
        runtrack[--runtrackpos] = 2;
    }
Label_0194:
    num4 = runstack[runstackpos++];
    this.Capture(2, num4, runtextpos);
    runtrack[--runtrackpos] = num4;
    runtrack[--runtrackpos] = 3;
    if (runtextpos >= runtextend)
    {
        goto Label_02F8;
    }
    runtextpos++;
    if (runtext[runtextpos] != '-')
    {
        goto Label_02F8;
    }
    num4 = runstack[runstackpos++];
    this.Capture(1, num4, runtextpos);
    runtrack[--runtrackpos] = num4;
    runtrack[--runtrackpos] = 3;
Label_0213:
    if (num4 != -1)
    {
        runtrack[--runtrackpos] = num4;
    }
    else
    {
        runtrack[--runtrackpos] = runtextpos;
    }
    if ((num4 = runstack[runstackpos++]) != runtextpos)
    {
        runtrack[--runtrackpos] = runtextpos;
        runtrack[--runtrackpos] = 4;
    }
    else
    {
        runstack[--runstackpos] = num4;
        runtrack[--runtrackpos] = 5;
    }
    if (((3 > (runtextend - runtextpos)) || 
    (runtext[runtextpos] != '-')) || 
    ((runtext[runtextpos + 1] != '%') || 
    (runtext[runtextpos + 2] != '>')))
    {
        goto Label_02F8;
    }
    runtextpos += 3;
    num4 = runstack[runstackpos++];
    this.Capture(0, num4, runtextpos);
    runtrack[--runtrackpos] = num4;
    runtrack[--runtrackpos] = 3;
Label_02EF:
    base.runtextpos = runtextpos;
    return;
Label_02F8:
    base.runtrackpos = runtrackpos;
    base.runstackpos = runstackpos;
    this.EnsureStorage();
    runtrackpos = base.runtrackpos;
    runstackpos = base.runstackpos;
    runtrack = base.runtrack;
    runstack = base.runstack;
    switch (runtrack[runtrackpos++])
    {
        case 1:
            runstackpos++;
            goto Label_02F8;

        case 2:
            runtextpos = runtrack[runtrackpos++];
            num4 = runtrack[runtrackpos++];
            if (num4 > 0)
            {
                runtrack[--runtrackpos] = num4 - 1;
                runtrack[--runtrackpos] = runtextpos - 1;
                runtrack[--runtrackpos] = 2;
            }
            goto Label_0194;

        case 3:
            runstack[--runstackpos] = runtrack[runtrackpos++];
            this.Uncapture();
            goto Label_02F8;

        case 4:
            runtextpos = runtrack[runtrackpos++];
            runstack[--runstackpos] = runtextpos;
            runtrack[--runtrackpos] = 5;
            if ((runtrackpos > 40) && (runstackpos > 30))
            {
                runstack[--runstackpos] = runtextpos;
                runtrack[--runtrackpos] = 1;
                runstack[--runstackpos] = runtextpos;
                runtrack[--runtrackpos] = 1;
                num4 = (num5 = runtextend - runtextpos) + 1;
                do
                {
                    if (--num4 <= 0)
                    {
                        goto Label_0161;
                    }
                    runtextpos++;
                }
                while (runtext[runtextpos] != '-');
                runtextpos--;
                goto Label_0161;
            }
            runtrack[--runtrackpos] = 6;
            goto Label_02F8;

        case 5:
            runstack[runstackpos] = runtrack[runtrackpos++];
            goto Label_02F8;
    }
    runtextpos = runtrack[runtrackpos++];
    goto Label_02EF;
}

Ответы [ 4 ]

5 голосов
/ 29 января 2011

Один вариант использования, который я вижу, - это автоматически сгенерированный код.Высокоуровневые конструкции потока управления просты для нас, людей, но не обязательно удобны в качестве вывода алгоритмов генерации кода.

И декомпилирован ли ваш код?В этом случае это может быть написано так, что декомпилятор не сможет справиться без использования goto s.На уровне IL все конструкции потока управления переводятся в goto s, и декомпилятор пытается угадать, что они были.И в сложных случаях он может не найти хороший способ представить его с помощью высокоуровневых конструкций и, следовательно, прибегает к goto s.

Другим вариантом использования являются конечные автоматы.У вас есть несколько состояний, и после каждого оно может перейти в новое состояние с помощью операторов goto.Парсеры - это обычное применение конечных автоматов.

3 голосов
/ 29 января 2011

ОК, это вопрос о сгенерированном коде , и он не имеет реального отношения к , почему C # имеет оператор goto.

И он не только генерируется, но и генерируется для конечного автомата (DFA).И даже рукописные конечные автоматы будут иногда использовать goto.

Чтобы ответить на заглавный вопрос: Нет , goto существует для обратной совместимости и для решения небольших угловых случаев, таких как вложенные циклы (выход из) и, например, конечные автоматы.Это мало или совсем не связано с производительностью.

1 голос
/ 29 января 2011

Вы видите goto в результате процесса декомпиляции. В этом случае отсутствуют переводы if / else-scopes. Процесс декомпиляции не является переводом 1-1, поэтому декомпилятор часто не может правильно его перевести. .Net активно использует goto, так как это единственный способ для MSIL / машинного кода перемещаться. На самом низком уровне он перемещается между адресами памяти, играющими со стеком.

Для нас, смертных разработчиков, можно избежать этого, поскольку у нас есть команды, которые могут делать то, что нам нужно. while (true) { } на самом деле просто переводится как «if (true) goto someaddress» и в конце области действия «goto that ifaddressupthere», так что при использовании goto напрямую производительность может быть очень низкой или совсем незначительной.

Я пишу некоторые вещи для переписывания ассемблера, используя Mono.Cecil , поэтому у меня есть некоторый опыт просмотра кода в Reflector и MSIL. Скомпилированный код в большинстве случаев довольно хорошо оптимизирован.

0 голосов
/ 29 января 2011

Есть несколько случаев, когда goto повышает читабельность кода, например разрыв из вложенных циклов.

Item item = null;
foreach(var a in A)
{
    foreach(var b in a.B)
    {
        if (b.foo == someCondition)
        {
            item = b.item;
            goto AfterLoop;
        }
    }
}
AfterLoop:

Без goto вам нужно много кодов if (hasFound) {break;}.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...