본문 바로가기
-----ETC-----/C++ 임베디드 시스템 프로그래밍 시리즈

[C++ 임베디드 시스템 프로그래밍 시리즈] Day 27: 임베디드 시스템의 메모리 최적화

by cogito21_cpp 2024. 8. 1.
반응형

메모리 최적화의 중요성

임베디드 시스템은 제한된 메모리 자원을 갖고 있기 때문에 메모리 사용을 최적화하는 것이 중요합니다. 메모리 최적화를 통해 시스템의 성능을 향상시키고, 안정성을 높일 수 있습니다. 오늘은 메모리 최적화를 위한 다양한 기법을 학습하겠습니다.

1. 데이터 타입 최적화

변수의 데이터 타입을 최적화하여 메모리 사용을 줄일 수 있습니다. 예를 들어, 필요한 경우 int 대신 byte 또는 char와 같은 작은 데이터 타입을 사용합니다.

 

예제: 데이터 타입 최적화

// 데이터 타입 최적화 전
int temperature = 25;
int humidity = 60;

// 데이터 타입 최적화 후
byte temperature = 25;
byte humidity = 60;

 

위 예제에서는 int 대신 byte를 사용하여 메모리 사용을 줄였습니다.

2. 배열과 문자열 최적화

배열과 문자열의 크기를 최적화하여 메모리 사용을 줄일 수 있습니다. 필요한 경우 정적 배열 대신 동적 메모리 할당을 사용하여 메모리 사용을 최적화할 수 있습니다.

 

예제: 문자열 최적화

// 문자열 최적화 전
char message[] = "This is a very long message that takes up a lot of memory";

// 문자열 최적화 후
const char* message = "Short message";

 

위 예제에서는 긴 문자열 대신 짧은 문자열을 사용하여 메모리 사용을 줄였습니다.

3. 동적 메모리 할당

동적 메모리 할당을 사용하여 필요한 시점에 메모리를 할당하고, 사용이 끝난 후 메모리를 해제하여 메모리 사용을 효율적으로 관리할 수 있습니다.

 

예제: 동적 메모리 할당

void setup() {
    Serial.begin(115200);

    // 동적 메모리 할당
    char* buffer = (char*)malloc(100 * sizeof(char));
    if (buffer == NULL) {
        Serial.println("Memory allocation failed");
        return;
    }

    // 메모리 사용
    strcpy(buffer, "Hello, world!");
    Serial.println(buffer);

    // 메모리 해제
    free(buffer);
}

void loop() {
    // 메인 루프는 비워둠
}

 

위 예제에서는 malloc 함수를 사용하여 동적으로 메모리를 할당하고, free 함수를 사용하여 메모리를 해제합니다.

4. 플래시 메모리 사용

프로그램 메모리(플래시 메모리)를 사용하여 상수를 저장함으로써 SRAM 사용을 줄일 수 있습니다.

 

예제: 플래시 메모리 사용

#include <avr/pgmspace.h>

// 플래시 메모리에 문자열 저장
const char message[] PROGMEM = "Hello, world!";

void setup() {
    Serial.begin(115200);

    // 플래시 메모리에서 문자열 읽기
    char buffer[20];
    strcpy_P(buffer, message);
    Serial.println(buffer);
}

void loop() {
    // 메인 루프는 비워둠
}

 

위 예제에서는 PROGMEM 키워드를 사용하여 문자열을 플래시 메모리에 저장하고, strcpy_P 함수를 사용하여 플래시 메모리에서 문자열을 복사해옵니다.

5. 메모리 풀 (Memory Pool) 사용

메모리 풀이란 고정된 크기의 메모리 블록을 미리 할당해두고, 필요할 때 블록을 할당하고 사용 후 반환하는 방식입니다. 메모리 풀을 사용하면 메모리 할당 및 해제의 오버헤드를 줄이고, 메모리 파편화를 방지할 수 있습니다.

 

예제: 메모리 풀 사용

다음 예제에서는 간단한 메모리 풀을 구현하여 작은 메모리 블록을 할당하고 해제하는 방법을 보여줍니다.

#include <Arduino.h>

// 메모리 풀 크기 및 블록 크기 설정
#define POOL_SIZE 10
#define BLOCK_SIZE 32

// 메모리 풀 선언
uint8_t memoryPool[POOL_SIZE][BLOCK_SIZE];
bool blockUsed[POOL_SIZE] = { false };

// 메모리 블록 할당 함수
void* allocateBlock() {
    for (int i = 0; i < POOL_SIZE; i++) {
        if (!blockUsed[i]) {
            blockUsed[i] = true;
            return (void*)memoryPool[i];
        }
    }
    return NULL; // 메모리 풀에 사용 가능한 블록이 없을 때
}

// 메모리 블록 해제 함수
void freeBlock(void* block) {
    for (int i = 0; i < POOL_SIZE; i++) {
        if (memoryPool[i] == block) {
            blockUsed[i] = false;
            return;
        }
    }
}

void setup() {
    Serial.begin(115200);

    // 메모리 블록 할당 및 사용 예제
    uint8_t* block1 = (uint8_t*)allocateBlock();
    if (block1 != NULL) {
        strcpy((char*)block1, "Hello, Memory Pool!");
        Serial.println((char*)block1);
        freeBlock(block1);
    } else {
        Serial.println("Memory allocation failed");
    }
}

void loop() {
    // 메인 루프는 비워둠
}

 

위 예제에서는 메모리 풀을 사용하여 고정된 크기의 메모리 블록을 할당하고, 사용 후 반환하는 방법을 구현했습니다.

마무리

오늘은 임베디드 시스템의 메모리 최적화를 위한 다양한 기법을 학습했습니다. 데이터 타입 최적화, 배열과 문자열 최적화, 동적 메모리 할당, 플래시 메모리 사용, 메모리 풀 사용 등의 기법을 통해 메모리 사용을 효율적으로 관리할 수 있습니다. 다음 단계에서는 저전력 설계 기법에 대해 알아보겠습니다.

질문이나 추가적인 피드백이 있으면 언제든지 댓글로 남겨 주세요.

Day 28 예고

다음 날은 "저전력 설계 기법"에 대해 다룰 것입니다. 저전력 설계 기법을 통해 임베디드 시스템의 전력 소비를 줄이고, 배터리 수명을 연장하는 방법을 학습하겠습니다.

반응형