Почему порядок вычисления параметров функции не указан в C++?

Стандарт не определяет порядок вычисления аргументов с этой строкой:

Порядок оценки аргументов не уточняется.

Что значит

Лучший код может быть сгенерирован при отсутствии ограничений на порядок вычисления выражений

означают?

Какой недостаток заключается в том, что все компиляторы просят оценить аргументы функции, например, слева направо? Какие виды оптимизации выполняют компиляторы из-за этой неопределенной спецификации?

2 ответа

Разрешение компилятору переупорядочивать оценку операндов добавляет больше возможностей для оптимизации.

Вот полностью составленный пример для иллюстрации.

Предположим, процессор может:

  • Выпуск 1 инструкции каждый цикл.
  • Выполните сложение в 1 цикле.
  • Выполните умножение в 3 цикла.
  • Может выполнять сложения и умножения одновременно.

Теперь предположим, что у вас есть вызов функции следующим образом:

foo(a += 1, b += 2, c += 3, d *= 10);

Если вы должны выполнить это слева направо на процессоре без OOE:

Cycle - Operation
0     -    a += 1
1     -    b += 2
2     -    c += 3
3     -    d *= 10
4     -    d *= 10
5     -    d *= 10

Теперь, если вы позволите компилятору переупорядочить их: (и сначала запустите умножение)

Cycle - Operation
0     -    d *= 10
1     -    a += 1, d *= 10
2     -    b += 2, d *= 10
3     -    c += 3

Итак, 6 циклов против 4 циклов.

Опять же, это полностью надумано. Современные процессоры намного сложнее. Но ты получил идею.

Вот простой пример. Предположим, у вас есть вызов функции следующим образом:

// assume that p is a pointer to an integer
foo(*p * 3, bar(), *p * 3 + 1);

Компилятору нужно разыменовать p дважды (и сделать некоторые вычисления на основе результата) и вызвать bar один раз. Если компилятор умен, он может изменить порядок оценки

int temp = *p * 3;
foo(temp, bar(), temp + 1);

Таким образом, он должен выполнить "разыменование, умножить на 3" только один раз. Это известно как устранение общего подвыражения.

Другие вопросы по тегам