C++ - Являются ли параметры const и переменные класса пессимизацией?
Я пытаюсь выяснить, когда const следует использовать при написании кода C++. Это все примеры пессимизации или так полезно писать код?:
Пример 1:
int findVal(const int OTHER_VAL) const
{
switch(OTHER_VAL)
{
case 1:
return 2;
default:
return 3;
}
}
Пример 2:
enum class MobType
{
COW, CHICKEN, DOG, PIG
};
class BaseMob
{
protected:
BaseMob(const MobType TYPE) : TYPE(TYPE) { }
const MobType TYPE;
};
Пример 3:
void showWorld(const World &world)
{
auto data = world.getData();
for (auto &i:data)
i.print();
}
2 ответа
Всякий раз, когда вы логически не изменяете значение или объект, вы должны сделать это const
, Логически я имею в виду не каждый раз, когда вам это технически разрешено, но каждый раз, когда это логично в контексте ваших функций, классов и кода.
Простым примером может быть простая функция "get", как видно в примере 1, эти функции не должны изменять состояние класса, и поэтому должны быть помечены как постоянные, поскольку это поможет документировать ваше намерение для пользователя, а также поможет вам обеспечить инвариантность класса.
Существуют ситуации, когда имеет смысл создать неизменный объект, как показано в примере 2. Мы не так часто видим их в C++, но многие другие языки часто их используют. Если он не добавляет никакого значения, чтобы иметь возможность изменять определенный элемент в течение времени жизни объекта, вы можете также сделать его константой.
Передача константных параметров ссылки дает вам преимущества производительности ссылки, но в то же время гарантирует, что исходный объект остается неизменным, что является отличной документацией для пользователя, но также позволяет выполнять некоторые оптимизации.
Упомянув все эти причины, есть и другие const
как кратко упомянуто в последнем абзаце, оптимизации. Когда компилятор знает, что что-то является постоянным и не изменяется, он может включить некоторые довольно умные оптимизации, не используйте const
хотя по соображениям производительности.
Это также то, почему работа вокруг константы (например) const_cast
литье, которое можно отбросить const
, может привести к некоторому нежелательному поведению. В качестве примера проверьте следующее:
#include <stdio.h>
static const int foo = 10;
int constsum(void) {
return foo + 5;
}
int main(int argc, char* argv[]) {
int a = constsum();
int* newFoo = const_cast<int*>(&foo);
*newFoo = 20;
int b = constsum();
printf("%d\n", a + b);
return 0;
}
Как видно из этого примера ( см. Код, выполняющийся здесь), это может не дать желаемого результата, так как результат кода в 30
быть напечатанным, а не как ожидалось 40.
Изучая произведенную сборку, мы видим, почему ( скомпилировано в сборку):
constsum():
mov eax, 15
ret
main:
mov eax, 30
ret
Компилятор просто вставляет значения, так как он может видеть, что они постоянны, он не заботится о том, чтобы const_cast
используется.
Так что const правильность и использование const
это ценный инструмент, который может повысить производительность и стабильность вашего кода, но также (и не забывать) помогает документировать ваш код.
Нет, это не так.
const
Локальные переменные с автоматическим хранением (включая аргументы функций) являются чисто синтаксическим сахаром, помогающим программистам-людям устанавливать правила для своего кода. Это не помогает оптимизатору вообще. Оптимизирующие компиляторы извлекают необходимое перемещение данных из источника C и оптимизируют его. Как правило, им все равно, если вы повторно используете одну и ту же переменную tmp для множества разных вещей или у вас есть 10 разных const tmp1 = a+10;
в той же функции.
И да, это относится к аргументам функции, переданным по значению; это локальные переменные с автоматическим хранением, передаваемые в регистрах или в стеке. И нет, это не означает, что вызывающая сторона может предположить, что функция не модифицировала стековую память, используемую для передачи аргументов, так что это также мало помогает оптимизатору. (Выполнение второго вызова функции с теми же аргументами все еще требует перезаписи аргументов в стек (если не все аргументы помещаются в регистры), потому что const
Аргумент arg не меняет того факта, что вызываемая функция "владеет" этим пространством стека и может использовать его как пустое место, как захочет.)
const
на статические / глобальные / ссылочные переменные помогает. static const int foo = 10;
может быть встроен как непосредственная константа вместо загрузки из памяти. (например add eax, 10
вместо add eax, [foo]
).
С помощью const
пометить метод класса как не меняющий членов класса также может помочь компилятору избежать повторной загрузки членов класса после вызова функции. (т.е. держать их вживую в регистрах). Это в основном применимо только в том случае, если компилятор не может увидеть определение функции, в противном случае хороший оптимизирующий компилятор может просто посмотреть на то, что делает вызываемая функция, и соответственно оптимизировать. (Пока это не в библиотеке Unix, где символьное расположение означает, что он не может предположить, что вызванная функция, которую он видит во время компиляции, будет вызвана после динамического связывания.)