C++

const 변수에 대한 최적화

Keisa 2024. 12. 17. 17:50

1. 불변성 기반 최적화

const 변수는 선언 이후 값이 변경되지 않는다는 점을 컴파일러가 확실히 알고 있습니다. 이 정보를 바탕으로 컴파일러는 다음과 같은 최적화를 수행할 수 있습니다.

1-1. 값의 인라인화

const 변수는 값이 불변이므로, 컴파일러는 변수 대신 그 값을 코드에 직접 인라인할 수 있습니다. 이를 통해 변수를 참조하는 과정이 제거되어 런타임 성능이 향상됩니다.

예시 코드:

const int VALUE = 10;

void printValue() {
    std::cout << VALUE << std::endl;
}
const int VALUE = 10; void printValue() { std::cout << VALUE << std::endl; }

컴파일 결과 (최적화 적용):

mov eax, 10   ; VALUE의 값이 바로 사용됨
call _ZNSolsEi ; std::cout에 10 출력
mov eax, 10 ; VALUE의 값이 바로 사용됨 call _ZNSolsEi ; std::cout에 10 출력

여기서 VALUE는 메모리에 저장되지 않고, 값 10이 직접 코드에 하드코딩되어 있습니다.


1-2. 불필요한 메모리 접근 제거

일반 변수의 값은 언제든 변경될 수 있기 때문에 컴파일러는 매번 메모리에 접근해야 합니다. 하지만 const 변수는 값이 절대 변경되지 않음을 보장하므로, 불필요한 메모리 읽기와 쓰기 작업을 생략할 수 있습니다.

일반 변수 예시:

int a = 5;
for (int i = 0; i < 100; ++i) {
    std::cout << a << std::endl;
}
int a = 5; for (int i = 0; i < 100; ++i) { std::cout << a << std::endl; }

여기서 a는 일반 변수이므로, 루프마다 a의 값을 읽습니다.

const 변수 예시:

const int a = 5;
for (int i = 0; i < 100; ++i) {
    std::cout << a << std::endl;
}
const int a = 5; for (int i = 0; i < 100; ++i) { std::cout << a << std::endl; }

컴파일러는 a가 변하지 않는다는 것을 알기 때문에, 루프마다 메모리를 읽지 않고 값 5를 그대로 사용합니다.


2. 코드 분석 및 최적화 기회 확대

const 변수의 불변성을 활용하면 컴파일러가 코드의 동작을 더 정확하게 예측하고 최적화를 적용할 수 있습니다.

2-1. 상수 전파(Constant Propagation)

컴파일러는 const 변수의 값을 상수로 치환하고, 그 값을 코드 전체에 전파할 수 있습니다. 이를 통해 불필요한 계산이 제거됩니다.

예시 코드:

const int x = 5;
int y = x + 2;
std::cout << y << std::endl;
const int x = 5; int y = x + 2; std::cout << y << std::endl;

컴파일 결과 (최적화 적용):

mov eax, 7   ; x + 2는 컴파일 타임에 계산됨
call _ZNSolsEi ; std::cout에 7 출력
mov eax, 7 ; x + 2는 컴파일 타임에 계산됨 call _ZNSolsEi ; std::cout에 7 출력

여기서 x + 2는 컴파일 시 바로 7로 계산되어 실행 시점에 불필요한 덧셈 연산이 사라집니다.


2-2. 죽은 코드 제거 (Dead Code Elimination)

const 변수를 기반으로 조건문이나 루프가 항상 참/거짓으로 평가되면, 해당 코드의 일부를 제거할 수 있습니다.

예시 코드:

const bool DEBUG = false;

if (DEBUG) {
    std::cout << "Debug mode" << std::endl;
}
const bool DEBUG = false; if (DEBUG) { std::cout << "Debug mode" << std::endl; }

컴파일 결과 (최적화 적용):

  • DEBUG가 false로 설정되어 있기 때문에 if 조건문과 그 내부 코드는 컴파일 단계에서 제거됩니다.

3. 함수 최적화와 호출 규칙

3-1. const 참조를 사용한 함수 최적화

const 변수는 값이 변경되지 않으므로, 불필요한 복사 연산을 줄이고 레지스터 최적화를 돕습니다.

예시:

void func(const std::string& str);

int main() {
    std::string text = "Hello";
    func(text);
}
void func(const std::string& str); int main() { std::string text = "Hello"; func(text); }
  • const std::string&는 복사가 일어나지 않으므로 메모리 낭비를 줄일 수 있습니다.
  • 컴파일러는 str이 절대 변경되지 않는다는 것을 알고 불필요한 로컬 복사본을 생성하지 않습니다.

4. 멀티스레딩에서 안전성 향상

const 변수를 사용하면 값이 변경될 가능성이 없기 때문에 멀티스레드 환경에서 동기화 비용이 줄어듭니다. 불변 데이터를 여러 스레드에서 동시에 읽어도 충돌이 발생하지 않습니다.


결론: 컴파일러 최적화 요약

const 변수를 사용하면 컴파일러가 다음과 같은 최적화를 수행할 수 있습니다:

  1. 값 인라인화: 메모리 접근 없이 직접 상수를 사용.
  2. 불필요한 메모리 접근 제거: 값이 불변이므로 불필요한 읽기/쓰기를 줄임.
  3. 상수 전파: const 값을 코드 전체에 전파하여 계산을 줄임.
  4. 죽은 코드 제거: 조건문이나 불필요한 코드를 제거.
  5. 레지스터 최적화: 불변 데이터는 레지스터에 저장할 수 있어 성능 향상.
  6. 동기화 비용 감소: 멀티스레딩 환경에서 안전성이 향상됨.

const 변수는 값이 불변임을 보장하기 때문에 컴파일러가 더 공격적인 최적화를 수행할 수 있도록 돕습니다. 결과적으로 성능이 향상되고 코드의 안전성과 가독성이 높아집니다.

'C++' 카테고리의 다른 글

STL 사용의 장점  (0) 2025.03.09
const 변수와 일반 변수  (0) 2024.12.17
std::bitset<'비트크기'>  (0) 2024.08.06
템플릿의 정의를 CPP에 쓸 수 없는 이유  (0) 2024.07.25
__declspec(dllexport)  (0) 2024.06.20