매크로는 컴퍼일러에게 코드의 특성을 알려주는 키워드이다. 따라서 기계어로 컴파일 과정에서 필요한 요소이고, 매크로 자체가 기계어 코드로 생성되지는 않지만 특정 코드를 제어하는데 사용한다.

파일 포함하기 편집

#include 편집

특정 프로그램 파일을 현재 위치에 첨부하여 하나의 파일처럼 컴파일한다. 보통 IDE 개발툴들은 보통 프로젝트(Project)라는 개념이 도입되어 있다. 한개의 프로젝트를 수행하기 위한 프로그램 파일 및 설정 등을 관리하고, 컴파일하여 실행 파일을 만들고 디버깅을 제공한다. 여러개의 파일을 프로젝트에 등록만 하면 서로 설정된 함수나 변수를 사용할 수 있다. 프로그램 파일이 개별적으로 저장되고 개별적으로 컴파일되어 오브젝트 파일을 생성한다. 서로 다른 함수의 함수나 기타를 활용하기 위해 함수형이나 변수형이나 선언 내용을 다른 파일 끼리 연결하는 방식이 필요하다. 이때 #include을 사용한다.

#include <stdio.h>

int main(void)
{
    printf("Hello, world!\n");
    return 0;
}

printf함수가 stdio.h 파일에 정의되어 있다. 따라서 printf함수의 모양을 알 수 있다. 만약 함수의 모양을 모른다면 호출이 불가능해진다.

조건부 컴파일하기 편집

매크로 #if, #ifdef, #ifndef, #else, #elif, #endif는 조건부 컴파일하도록 정의한다. 따라서 조건이 맞지 않으면 컴파일에서 제거 된다. 즉, 조건이 맞지 않으면 소스 자체가 없는 것과 같은 효과가 있다.

#if ~ [#elseif] ~ [#else] ~ #endif 편집

#if VERBOSE >= 2
  print("trace message");
#endif

VERBOSE가 정의 된 값에 따라 print 함수를 제거할 수 있다. 조건 판단은 C언어 조건과 같이 예에서 VERBOSE가 2보다 크거나 같으면 컴파일한다. 이 경우 VERBOSE 값이 미리 정의되어 있어야 조건에서 참이 가능하다.

프로그램 코드 작성 시, 임의 영역을 다양한 상황에 맞게 컴파일에서 제거할 수 있다.

복합 정의와 조건부 편집

#ifdef __unix__ /* __unix__ is usually defined by compilers targeting Unix systems */
# include <unistd.h>
#elif defined _WIN32 /* _Win32 is usually defined by compilers targeting 32 or 64 bit Windows systems */
# include <windows.h>
#endif

이 경우 OS가 UNIX 또는 윈도우 환경에 따라 다른 파일을 포함 시킬 수 있다. __unix__가 정의되어 있다면, unistd.h 파일을 첨부하고 그렇지 않고 _WIN32가 정의되어 있다면 windows.h 파일을 첨부한다. __unix__나 _WIN32의 정의는 다양한 방식으로 정의 할 수 있다.

매크로 #ifdefined와 연결하면 다음과 같이 복잡한 조건부 컴파일이 가능하다.

#if !(defined __LP64__ || defined __LLP64__) || defined _WIN32 && !defined _WIN64
	// we are compiling for a 32-bit system
#else
	// we are compiling for a 64-bit system
#endif

#error을 다음과 사용하여 컴파일 실패하게 만들 수 있다.

#if RUBY_VERSION == 190
# error 1.9.0 not supported
#endif

매크로 정의와 확장 편집

#define #undef 형식 편집

#define은 특정 숫자, 함수, 프로그램 블록을 다른 형태로 변환 지정한다. 다양한 선언이 가능하다. 조건은 선언 된 그 줄에 만 적용된다. 다음 줄로 연결하려면 '\'을 사용하여 한 줄처럼 코딩하면 된다.

프로그램 파일 내에서 #define을 사용하는 것이 일반적이지만 makefile이나 기타 IDE 툴의 설정을 통해서도 정의 할 수 있다. 따라서 컴파일되는 파일에 없다고 무조건 선언이 되지 않았다는 보장이 없다. 남의 프로그램을 사용할 때는 주의가 필요하다.

이렇게 정의 내용을 해제하려면 #undef을 사용하면 정의 된것이 없었던 것이 된다. 만약 어떤식으로든 이미 특정 정의가 선언되어 있다면 다시 설정할 때, #undef을 이용해 제거하고 다시 설정하면 된다.

매크로 선언은 대상변환형(Object-like macros)[1]유사함수변환형(function-like macros)[2] 가 있다.

대상변환형은 인수가 없고, 유사함수변환형은 함수와 유사하게 정의 :

#define <identifier> <replacement token list>                    // object-like macro
#define <identifier>(<parameter list>) <replacement token list>  // function-like macro, note parameters

정의 된 매크로는 "#undef"을 사용하여 무효화 :

#undef <identifier>                                              // delete the macro

사용 예 편집

대상변환형 :

#define PI 3.14159

코딩 중 PI 토큰이 나오면 바로 3.14159로 바꾸어 컴파일한다. 단순한 문자 교환이 이루어지는 것이다. 이렇게 하면 장점은 특정 숫자의 의미가 정확히 와닿지 않는 것을 방지할 수 있다.

     #define BUFFER_SIZE 1024

     foo = (char *) malloc (BUFFER_SIZE);
     for (int count = 0;count < BUFFER_SIZE;count++)
        // ...

코드 내에 정의 된 BUFFER_SIZE는 1024 바꾸어 컴파일한다. 이것은 다음과 같이 코딩한 것과 같다 :

     foo = (char *) malloc (1024);
     for (int count = 0;count < 1024;count++)
        // ...

코딩이 많아 질 수록 1024가 갖는 의미가 혼돈 스러울 수가 있다.

     #define NUMBERS 1, \
                     2, \
                     3
     int x[] = { NUMBERS }; // ==> int x[] = { 1, 2, 3 }; 와 같다.

#define 한 줄에 정의된 것만 취급한다. 다음 줄에 계속 연결할 때는 위와 같이 \ 을 사용한다.

     foo = X;
     #define X 4
     bar = X;

매크로 정의는 정의된 이후부터 적용된다. 따라서 위 코드는 :

     foo = X;
     bar = 4;

토큰 교환이 bar 변수에만 적용된다.

유사함수변환형:

#define RADTODEG(x) ((x) * 57.29578)

특별한 매크로와 지시문 편집

## 토큰 연결 연산자 편집

토큰 연결 연산자는 두개의 토큰을 하나의 토큰으로 연결한다. 예를 들어:

#define DECLARE_STRUCT_TYPE(name) typedef struct name##_s name##_t

DECLARE_STRUCT_TYPE(g_object); // 출력 결과는 typedef struct g_object_s g_object_t;

기타 편집

#pragma 편집

컴파일러에게 특성 옵션이나 라이브러리 형태등을 지정할 수 있다. 다양하므로 한마디로 정의 할 수 없다. 여러 가지 환경에 적응할 수 있도록 컴파일러 마다 그리고 CPU 및 시스템 특성에 맞는 것들이 추가되거나 삭제된다.

같이 보기 편집

각주 편집

  1. “Object-like macros”. 2012년 12월 13일에 원본 문서에서 보존된 문서. 2012년 11월 25일에 확인함. 
  2. “Function-like macros”. 2012년 12월 13일에 원본 문서에서 보존된 문서. 2012년 11월 25일에 확인함. 

외부 링크 편집