단항 연산자 중에서 증감 연산자 ++,--가 있고 변수 앞에 붙이면 전치 증감 연산자(++i,--i)라고 하고 변수 뒤에 붙이면 후위 증감 연산자(i++,i--)라고 합니다.

•전위 증감 연산자는 실행문에서 값이 먼저 증가/감소해서 적용됩니다.

1. int i = 1; 
2. int j = ++i;
j = 2 , i = 2 가 된다.

•후위 증감 연산자는 실행문에서 값이 후에 증가/감소해서 적용됩니다.
    1. int i = 1; 
    2. int j = i++;
    j = 1 , i = 2 가 된다.


보통  for문에 자주 사용하고 전위,후위 연산자 모두 같은 결과를 나타내지만 내부적으로 동작 원리가 다릅니다.

• 전위 증감 연산자(++i,--i)
    1. i = i + 1; 
    2. return i;


•후위 증감 연산자(i++,i--)
    1. const int temp = i;
    2. i = i + 1;
    3. return temp;


전위 증감 연산자는 변수 자체의 값을 1 증가/감소해서 반환하지만 후위 증감 연산자는 다른 변수에 현재 변수 값을 저장하고 변수 값을 1 증가/감소하고 이전에 저장된 값을 반환합니다.


보통 for문의 증감식에 증감 연산자를 사용하게 되는데 컴파일러에서 최적화를 하기 때문에 차이가 없을 수도 있지만 최적화를 하지 않을 경우에는 전위 증감 연산자를 사용하면 후위 증감 연산자 보다 성능상 이득이 있을 수 있습니다.


연산자 오버로딩으로 증감 연산자를 만든 경우에 후위 증감 연산자는 클래스를 복사해야되므로 성능이 더 떨어지게 됩니다.


결론은 증감 연산자를 사용할 때 전위나 후위 모두 사용해도 될 경우에는 전위 증감 연산자를 사용하면 성능상 이득을 볼 수 있습니다.

'개발지식창고 > C' 카테고리의 다른 글

비트 시프트 연산자를 이용한 프로그램  (0) 2011.08.31
define함수 대신 template를 쓰자  (0) 2010.09.20
define 대신 inline을 쓰자  (0) 2010.09.20
C 문자열 함수 정리  (0) 2010.09.20
BASE64 Encode/Decode  (0) 2010.09.20
Posted by 모과이IT
,
비트 시프트 연산자를 이용한 프로그램
>>(오른쪽 시프트) 연산은 2로 나눈 효과를, <<(왼쪽 시프트) 연산은 2를 곱한 효과를 줌

#include <stdio.h>
#include <stdlib.h>
int main(void)
{
    int a=16, b=2;
    char bin[17];
    printf("변수 a의 값 : %3d\n", a);
    printf("a>>2 (10진수) : %3d\n", a>>2);
    printf("변수 b의 값 : %3d\n", b);
    printf("b<<2 (10진수) : %3d\n", b<<2);
    printf("\n");
    itoa(a, bin, 2);
    printf("a의 2진수 : %8s\n", bin);
    itoa(a>>2, bin, 2);
    printf("a>>2 결과 : %8s\n", bin);
    itoa(b, bin, 2);
    printf("b의 2진수 : %8s\n", bin);
    itoa(b<<2, bin, 2);
    printf("b<<2 결과 : %8s\n", bin);
    return 0;
}

// 실행 결과

'개발지식창고 > C' 카테고리의 다른 글

전위 후위 증감 연산자, ++i, i++ 차이  (0) 2012.09.28
define함수 대신 template를 쓰자  (0) 2010.09.20
define 대신 inline을 쓰자  (0) 2010.09.20
C 문자열 함수 정리  (0) 2010.09.20
BASE64 Encode/Decode  (0) 2010.09.20
Posted by 모과이IT
,

#define max(a,b) ((a) > (b) ? (a) : (b))

일 경우 우선 모든 인자를 괄호로 묶어야한다. 그렇지않으면 다른 문제가 생길수 도 있다.

또한 신기한상황을 보면

int a=5, b = 0;

max(++a,b);                     //a가 두번 증가한다.

max(++a,b+10);                //a가 한번 증가한다.

이유는 알것이다.

또한 서로 다른 여러가지 데이터 타입이 들어갈경우 예외 상황이 발생한다.

위의것을 template로 변형하면

template<class T>

inline const T& max ( const T& a, const T& b)

{return a > b ? a : b; }

이러한 template는 모든 타입의 함수를 생성한다.

이렇게 될경우 여러가지 타입을 쓸수있다.


'개발지식창고 > C' 카테고리의 다른 글

전위 후위 증감 연산자, ++i, i++ 차이  (0) 2012.09.28
비트 시프트 연산자를 이용한 프로그램  (0) 2011.08.31
define 대신 inline을 쓰자  (0) 2010.09.20
C 문자열 함수 정리  (0) 2010.09.20
BASE64 Encode/Decode  (0) 2010.09.20
Posted by 모과이IT
,
* inline함수
    - 매크로함수와 그냥 함수의 중간적인 성격을 띤 함수이다.


    - 일반적인 함수를 사용하면 필요할 때마다 호출하여 사용하므로 실행파일의 크기를

     줄일 수 있으나 제어권의 이동이 심하므로 실행 속도가 느려진다.


     - 매크로 함수는 컴파일시 전개되어 치환되므로 제어권 이동이 발생하지 않으므로

     속도가 빠르지만 데이터형 지정을 할 수가 없고 실행파일의 용량이 커지게 된다.


    - 인라인 함수는 컴파일시 통째로 매크로처럼 전개되어 속도면에서 큰 이득을 얻는다.

     그리고 데이터형 체크를 할 수 있는 함수이다.
     용량이 커지는 단점이 있긴 하지만 간단한 작업을 하는 함수의 경우에 이용하면 매우

     편리하다.


 * inline 함수의 제약조건
    - inline 함수 내에서는 루프문(do whie, while, for), switch, goto문을 사용할 수 없다.
    - inline 함수호출시 호출되기 전에 먼저 inline 함수가 정의되어 있어야 한다.
    - inline 함수 내에서 재귀호출을 할수 없다.
    - inline 함수는 한 수식 내에서 두 번이상 호출될수 없다.
    - 함수 포인터로 inline 함수의 주소를 취할 수 없다.
    - inline 함수는 호출방식이 아니라 치환전개방식이기 때문이다.




1. 이 항목의 주요는 "전처리기 보다는 컴파일러를 선호 한다" 일 것이다!

#define ASPECT_RATIO (1.653)              // #define 보다는

[문제]

심볼 명 ASPECT_RATIO를 컴파일러는 전혀 알수 없다. ( 전처리기에서 처리를 하므로 ) 결과적으로 ASPECT_RATIO 란 이름은 심볼 테이블에 들어 가지 않는다.

컴파일 시 에러가 발생하면, 컴파일러는 "1.653" 을 언급하지 ASPECT_RATIO 를 언급하지 않는다, 따라서 작성자는 1.653이 어디에서 왔는지 알 수 없게 된다. 이것은 혼란을 야기 한다.

[해법]

const double ASPECT_RATIO = 1.653 ;    // const 를 쓰는 것이 좋다.

이렇게 const 를 사용하면, 컴파일시 에러가 발생해도, 즉시 무엇이 문제 인지 판단할 수 있다. (컴파일러가 일러 준다. )

2. 이 항목의 주요는 "#define 보다는 inline을 사용하자!" 이다.

#define max( a, b ) ( ( a ) > ( b ) ? ( a ) : ( b ) )  // 두 값중 최대값을 리턴한다.

[문제]

max( ++a, b ) ;        // a 가 두번 증가 한다.

max( ++a, b + 10 ) ; // a 가 한번 증가 한다.

위 매크로 함수는 max 내부에서 일어나는 그 무엇과 비교를 하느냐에 따라 값이 달라지는, 넌센스를 갖고 있다.

[해법]

inline int max ( int a , int b ) { return a > b ? a : b ; } // int 버전의 한계성이 있다.

한계를 극복하려면, 템플릿을 사용하면 된다.

template<class T>

inline const T& max ( const T& a, const T& b ) { return a > b ? a : b ; }

템플릿 버전은 모든 한계를 극복하고 모든 종류의 함수를 생성한다.


'개발지식창고 > C' 카테고리의 다른 글

비트 시프트 연산자를 이용한 프로그램  (0) 2011.08.31
define함수 대신 template를 쓰자  (0) 2010.09.20
C 문자열 함수 정리  (0) 2010.09.20
BASE64 Encode/Decode  (0) 2010.09.20
문자열 처리 함수들  (0) 2010.08.22
Posted by 모과이IT
,

스트링 함수에 대해서 정리하였습니다. 여기에서 빠진 함수가 있거나
잘못된 내용이 있다면 연락 주시기 바랍니다.
                    Written by LHC ( s1662543@kiscos.sarang.net)
 
    
               - 스트링 함수 v1.2-

1. 종류

     int strcasecmp(const char *s1, const char *s2);
     char *strcat(char *dest, const char *src);
     char *strchr(const char *s, int c);
     int strcmp(const char *s1, const char *s2);
     int strcoll(const char *s1, const char *s2);
     char *strcpy(char *dest, const char *src);
     size_t strcspn(const char *s, const char *reject);
     char *strdup(const char *s);
     char *strfry(char *string);
     size_t strlen(const char *s);
     char *strncat(char *dest, const char *src, size_t n);
     int strncmp(const char *s1, const char *s2, size_t n);
     char *strncpy(char *dest, const char *src, size_t n);
     int strncasecmp(const char *s1, const char *s2, size_t n);
     char *strpbrk(const char *s, const char *accept);
     char *strrchr(const char *s, int c);
     char *strsep(char **stringp, const char *delim);
     size_t strspn(const char *s, const char *accept);
     char *strstr(const char *haystack, const char *needle);
     char *strtok(char *s, const char *delim);
     size_t strxfrm(char *dest, const char *src, size_t n);
     char *index(const char *s, int c);
     char *rindex(const char *s, int c);
    
2. 설명
C에서의 string은 단순히 0바이트로 끝나는 문자들의 배열이다.
string에 관한 모든 타입과 선언은 string.h에 되어있다.
string 처리를 위해 필요한  str1의 첫번째 주소를 알기위해선 &str1[0]대신
str1이라고만 하면 된다. C에서 배열의 변수명은 배열의 첫번째 주소를 가르킨다는
것은 이글을 보는 사람이라면 모두 알고 있을 것이다. 배열역시 내부적으로는
포인터로써 처리된다. 즉
str1[i]가 있다면 이의 실제 처리는 다음과 같이 된다. *(str1 + i)
함수의 인자로써 배열을 주는 경우가 종종있는데 이는 좋은 코딩이 아니다. 왜냐하면
인자로 넘겨주는 배열은 위와같이 다시 포인터로 변환되는 과정을 거쳐야 하기때문에
코드의 효율성을 떨어뜨린다.
int func(char []) -> int func(char *)
위와 같이 한다하여 배열과 포인터가 결코 같다고 생각해선 안된다.
char i[10]; 하면 char 메모리를 10만큼 할당하는 것이지만 char *i; 하면 단지
메모리에 있는 문자를 가르키는 번지만이 할당된다. 다시 위의 함수의 인자를 넘겨주
는것에 대해 살펴보면 이와같은 차이에도 불구하고 왜 저렇게 해야하는지 의문이
들겠지만 함수에서는 인자로 값을넘겨줄때 배열의 모든값을 넘겨줄수가 없다.
따라서 그 번지를 넘겨줘야 되는데 어차피 같은 거라면 배열을 써서 다시 포인터로
변환되는 과정을 거칠필요가 없다는 것이다.
여기에서 더 깊이 들어가는것은 이 문서의 범위를 벗어나므로 더욱 자세한 사항은
포인터와 배열에 대한 책을 참고하기 바란다.(http://pw2.netcom.com/~tjensen/ptr/pointers.htm)
string 를 처리할때 가장많이 사용하는 함수로 strlen이 있는데 이는 아주 유용하다.
strlen은 문자열의 길이를 리턴해준다(null 문자는 길이에서 제외된다).
다시말하면 null 문자의 offset을 린턴하는 것이다. 보통 문제가 되는 것은 sizeof
와 이거와의 혼동이다.

char a[1024] = "하하하 호호호 히히히 메롱"; 있다고 하면
sizeof(a) 는 1024이고 strlen(a)는 25이다.

앞으로 자주보게 될 size_t는 unsigned int형이다.
스트링 처리를 함에 있어 그 정확한 길이를 알고 싶은 경우가 많은데
거의 대부분의 해답은 strlen()가 된다. 그것으로 해결이 안되는 경우는
함수의 리턴값을 통해서 알 수 있는 경우도 있다. 예를들면 read()나 write()의
경우에는 리턴값이 읽거나 쓴 길이이다.

2.1 복사 함수

      void *memcpy(void *dest, const void *src, size_t n);

이는 src가 가르키는 메모리 영역으로 부터 n 바이트를 dest 영역으로 복사한다.
주의할 점은 두개의 영역이 겹쳐있으면 안된다. 만약 그런경우가 필요하다면
위해선 memmove를 사용하라.
리턴값은 dest를 가르키는 포인터

     void *memmove(void *dest, const void *src, size_t n);

이는 src가 가르키는 메모리 영역으로 부터 n 바이트를 dest영역으로 복사하는데
두 메모리 영역은 겹쳐져도(overlap) 된다.
리턴값은 dest를 가르키는 포인터

     void *memccpy(void *dest, const void *src, int c, size_t n);

n 바이트를 복사하는데 문자 c를 만나면 중지한다.
리턴값은 dest에서 c문자 다음위치를 가르키는 포인터. 실패는 NULL

     void *memset(void *s, int c, size_t n);
s가 가르키는 곳을 c로서 n바이트 만큼 채운다.
리턴값은 s를 가르키는 포인터이다.

     char *strcpy(char *dest, const char *src);
src가 가르크는 메모리 영역의 내용을(널문자 포함) dest가 가르키는 메모리 영역으로
복사한다. 주의할 점은 dest는 복사되는 문자에 대해 모두 저장할 수 있어야한다.
리턴값은 dest를 가르키는 포인터이다.

     char *strncpy(char *dest, const char *src, size_t n);
strcpy와 같고 지정된 n 바이트만을 복사한다. 널문자는 복사되지 않는다.
만약 src가 가르키는 곳의 문자열 길이가 n보다 작다면 로써 패딩되어 복사된다.
strncpy를 사용하는 것이 strcpy에서
발생할 수있는 버그를 예방하는 것이긴 하지만 보통 이것은 수행속도가 느리다.
왜냐하면 상대적으로 작은 src를 상대적으로 큰 버퍼에 복사하는 과정에서 버퍼의
나머지영역을 널문자로 채우게 되는데 여기에서 상당한 시간을 보내기 때문이다.

     char *strcat(char *dest, const char *src);
src가 가르키는 메모리 내용을 dest가 가르키는 곳의 끝인 '' 위치부터 붙여
넣는다(널문자까지). overlap은 허용되지 않으며 dest는 부여지게 될 string까지
커버할 수있는 메모리를 가지고있어야한다.
리턴값은 dest를 가르키는 포인터이다.
strcat는 다음과 같다.

     char *
     strcat (char *to, const char *from)
    {
       strcpy (to + strlen (to), from);
       return to;
    }
    
     char *strncat(char *dest, const char *src, size_t n);
strcat와 같으며 src의 n byte만을 복사한다.
그런데 끝에 널문자까지 복사되기때문에 실제 복사되는 크기는 n + 1이 됨을
주의해야 한다.
src가 n보다 작다면 널문자로 패딩된다.
strncat의 리턴값은 a[0]에 대한 번지값이고 overlap은 허용되지 않는다.

       char *strdup(const char *s);
string duplicate. s의 문자열을 복사하고 그 문자열을 가르키는 포인터를
리턴한다. 새로운 문자열을 malloc에 의해서 메모리를 할당 받고
free를 통해서 해제할 수있다. 만약 할당할 메모리가 없다면 널 포인터를
리턴한다.
     char *name;
     name = strdup("LHC");
     free (name);

       char * strndup (const char *s, size_t size)
s에서 size 만큼의 문자열을 복사하고 복사된 문자열을 가르키는 포인터를 리턴한다.
size가 문자열 s보다 작다면 size만큼 복사하고 끝에 널포인터를 넣어준다.

2.2 비교 함수

     int memcmp(const void *s1, const void *s2, size_t n);
s1과 s2의 처음 n 바이트를 비교한다. s1이 s2보다 크면 0보다 큰값을 같으면
0을 작으면 음수값을 리턴한다.
임의의 배열에서 두개가 같은지 않같은지를 테스트하는데 이를 사용하는것은
아주 유용하다. 하지만 바이트 수가 다른 배열상의 byte-wise ordering 비교를
하는것은 의미없는 짓이다. 예를들면 실수형 포인터로 구성된 메모리상의
1 byte-wise ordering은 실수형 포인터 변수들사이의 관계에 대해 어떤것도
말해주지 않는다. 또다른 주의해야 할 사항으로 holes를 포함하는 객체들을
비교할 때이다. holes는 패딩하여 생기는 메모리 영역을 말하는데 패딩이란
예를들어 1byte씩 주고받는 상황이 있다고 할때 어떤 함수에서 5bit만을 리턴한다면
그를 규정에 맞게 주고 받기 위해선 8bit로 만들어야 한다. 이때 모자라는 3비트를
널로 채우든지 뭐로 채우든간에 이렇게 채우는 행위를 일컷는 용어이다.
이렇게 패딩된 holes의 내용은 무엇이 들어 갈지 알 수없기때문에 byte-wise 비교
수행시 문제를 일으킬수 있다. 따라서 결과에 대해 확실히 하려면 component-wise
비교를 해야한다.
struct foo
{
     unsigned char tag;
     union
     {
            double f;
            long i;
            char *p;
      } value;
};
즉 위와 같은 foo의 멤버 데이터를 비교하기 위해선 memcpy를 사용하는것보다
다른 특별한 함수를 사용하는것이 더 낫다.

     int strcmp(const char *s1, const char *s2);
s1과 s2를 비교하고 리턴값은 memcmp와 같다.
strcmp는 다음과 같이 이루어져있다.
int
strcmp(s1, s2)
register const char *s1, *s2;
{
        while (*s1 == *s2++)
               if (*s1++ == 0)
                        return (0);
             return (*(const unsigned char *)s1 - *(const unsigned char *)(s2 - 1));
                                        }
     int strncmp(const char *s1, const char *s2, size_t n);
s1의 처음 n 바이트만을 비교하고 나머지는 strcmp와 같다.
다음은 아스키 코드를 기반으로 하는 시스템에서의 결과이다.
strcmp ("hello", "hello")
=> 0    /* 같으므로 리턴값은 0 */
strcmp ("hello", "Hello")
=> 32   /* h와 H의 아스키 코드 차는 32 */
strcmp ("hello", "world")
=> -15  /* h와 w의 아스키 코드 차는 15 */
strcmp ("hello", "hello, world")
=> -44  /* 널문자와 ',' 와의 아스키코드 차는 44 */
strncmp ("hello", "hello, world", 5)
=> 0    /* 처음 5바이트는 같다 따라서 리턴값은 0*/
strncmp ("hello, world", "hello, stupid world!!!", 5)
=> 0    /* 처음 5바이트는 같다 따라서 리턴값은 0 */

       int strcasecmp(const char *s1, const char *s2);
strcmp와 같다.

       int strncasecmp(const char *s1, const char *s2, size_t n);
strncmp와 같다.

2.3비교 정열을 위한 함수 ( collation function )
문자를 비교나 정열하기 위해선 사용하는 문자의 코드값을 알아야 한다. 예를 들면
알파벳의 경우 a는 0x61번의 아스키 코드를 갖는다. 이를 통하여 우리는
문자의 순서를 알수있다. 그런데 문제는 비영어권에서이다. 그들의 문자에 대한
코드값이 필요한다 일반적인 프로그램에서는 그에 대한 정보를 알수 없기때문에
정열을 할수없게된다. 그래서 필요한 것이 로케일이란 것인데 이것은 사용자가
사용하는 나라의 문자에 관한 정보를 가지고있다. 그래서 프로그램은 그 로케일을
참고하여 정열이나 기타 그 나라 문자에 관한 처리를 할수있는 것이다.
이렇게 현재의 로케일에 맞게 비교 해주는 함수로 strcoll과 strxfrm이 있다.
이들 함수들은 로케일 category중 LC_COLLATE값을 사용한다.
표준 C에서 strcoll의 비교 방식은 strcmp와 동일하다.

     int strcoll(const char *s1, const char *s2);
현재의 locale를 사용하여 두 문자열을 비교한다. 리턴값은 memcpy와 같다.
(예제)
qsort를 이용한 strcoll 사용예제.
int
compare_elements (char **p1, char **p2)
{
  return strcoll (*p1, *p2);
  }

/* This is the entry point---the function to sort
   strings using the locale's collating sequence. */

void
sort_strings (char **array, int nstrings)
{
  /* Sort temp_array by comparing the strings. */
    qsort (array, sizeof (char *),
             nstrings, compare_elements);
}

size_t strxfrm(char *dest, const char *src, size_t n);
strxfrm은 src 스트링을 어떤 폼으로 변환시키는데 strxfrm으로 변환된 두개의
스트링을 strcmp 처리를 하면 변환되기전의 두개의 스트링을 strcoll 처리한것과
같은 결과를 갖게 된다. 변환된 n개 만큼의 문자들을 dest에 쓴다. 변환은
LC_COLLATE에 설정되 있는 현재 로케일을 따른다. overlap은 허용되지 않는다.
리턴값은 전체 변환된 문자의 바이트 수이고(는 제외) n에 영향 받지 않는다.
만약 변환된 문자열의 길이가 n과 같거나 크다면 dest에 들어간 내용은 정확하지
않음을 주의해야 한다. 정확한 dest를 얻으려면 다시한번 더큰 사이즈로 strxfrm을
호출해야한다.  만약 n이 0이면 어떤 문자도 dest에 복사하지 않고 단지 전체
변환된 문자열의 길이만을 리턴하고 dest의 사이즈에도 전혀 영향 받지안기 때문에
변환된 문자의 길이를 알때 유용하게 사용할 수있을것이다. 다음과 같이 하면 된다.
1 + strxfrm(NULL, "변환할 문자열", 0)
LC_COLLATE는 정규식에서 검색이나 스트링 정렬을 위해 사용되는 로케일 값이다.
로케일 category에 대해서 더 자세한 사항을 보려면 man setlocale로 확인하기
바란다. setlocale은 프로그램에서 사용할 로케일을 설정하는 함수이다.
현재 자신의 운영체제에서 사용중인 로케일을 확인하려면 locale란 명령을
주면된다.
[root@LHC rc.d]# locale
LANG=ko_KR.eucKR
LC_CTYPE="ko_KR.eucKR"
LC_NUMERIC="ko_KR.eucKR"
LC_TIME="ko_KR.eucKR"
LC_COLLATE="ko_KR.eucKR"
LC_MONETARY="ko_KR.eucKR"
LC_MESSAGES="ko_KR.eucKR"
LC_ALL=ko_KR.eucKR
[예제]
다음은 strcoll의 예제와 같은 역할을 하는 소스인데 스트링 변환을 1번만 하기
때문에 효율적이고 훨씬 빠르다. 여러번 비교를 할때 유용할 것이다.
struct sorter { char *input; char *transformed; };

/* This is the comparison function used with qsort
   to sort an array of struct sorter. */

   int
   compare_elements (struct sorter *p1, struct sorter *p2)
   {
     return strcmp (p1->transformed, p2->transformed);
   }

/* This is the entry point---the function to sort
   strings using the locale's collating sequence. */

   void
   sort_strings_fast (char **array, int nstrings)
   {
         struct sorter temp_array[nstrings];
         int i;

/* Set up temp_array.  Each element contains
   one input string and its transformed string. */
       for (i = 0; i < nstrings; i++)
       {
                 size_t length = strlen (array[i]) * 2;
                 char *transformed;
                 size_t transformed_lenght;

                 temp_array[i].input = array[i];

/* First try a buffer perhaps big enough.  */
                 transformed = (char *) xmalloc (length);

/* Transform array[i].  */
            transformed_length = strxfrm (transformed, array[i], length);

/* If the buffer was not large enough, resize it
   and try again.  */
            if (transformed_length >= length)
            {
/* Allocate the needed space. +1 for terminating
   NUL character.  */
                      transformed = (char *) xrealloc (transformed, transformed_length + 1);

/* The return value is not interesting because we know
   how long the transformed string is.  */
                      (void) strxfrm (transformed, array[i], transformed_length + 1);
          }

               temp_array[i].transformed = transformed;
          }

/* Sort temp_array by comparing transformed strings. */
        qsort (temp_array, sizeof (struct sorter), nstrings, compare_elements);
/* Put the elements back in the permanent array
   in their sorted order. */
       for (i = 0; i < nstrings; i++)
           array[i] = temp_array[i].input;
/* Free the strings we allocated. */
       for (i = 0; i < nstrings; i++)
        free (temp_array[i].transformed);
}

2.4 검색함수
       void *memchr(const void *s, int c, size_t n);
s가 가르키고 있는 문자열에서 n번째 이내에서 처음 나오는 c를 찾는다.
char를 찾는데 int c로 선언되 있음에 의아할 지도 모르겠지만 c는 찾는 과정에서
unsigned char로 변환된다. 리턴값은 찾았다면 문자가 위치한 곳의 포인터를
못찾으면 널포인터를 리턴한다.

       char *strchr(const char *s, int c);
s가 가르키고 있는 문자열에서 c를 찾는다. 종료문자인 널문자 역시 스트링의
부분으로써 간주된다. 따라서 C를 널문자로 줌으로써 문자열의 끝 번지를
알아내는데 이용할 수있다. 리턴값은 찾았다면 문자가 위치한 곳의 포인터를
못찾으면 널포인터를 리턴한다.
[예제]
strchr ("hello, world", 'l')
     => "llo, world"
strchr ("hello, world", '?')
     => NULL
     
      char *strrchr(const char *s, int c);
s가 가르키는 문자열에서 마지막 발생한 c를 찾는다.
리턴값은 찾으면 마지막 발생한 c의 주소를 못찾으면 널을 리턴한다.
널문자 역시 스트링의 일부로 간주된다.
[예제]
strrchr ("hello, world", 'l')
     => "ld"

       size_t strspn(const char *s, const char *accept);
이 함수는 s가 가르키는 문자열에서 accept로 구성된 문자가 안나올때 까지
문자열의 길이를 계산한다. 리턴값은 그 길이이다.
strspn은 string span이라 읽는다.
예를 들면
strspn ("hello, world", "abcdefghijklmnopqrstuvwxyz")
      => 5
accept로 구성된 문자가 안나올때까지의 길이는 5이다. ','는 accept에
정의되어 있지 않기 때문이다. 만약 accpet로 ", abcdefghijklmnopqrstuvwxyz"
해주면 13이라고 문자열 전체 길이를 출력할 것이다. accept 문자의 순서는 어떻게
되든 상관이 없다.

       size_t strcspn(const char *s, const char *reject);
string complement span 이라 읽으며 strspn의 반대이다. 좀 헤깔릴지도 모르겠는데
reject로 구성된 문자가 나올때까지의 길이를 리턴해 준다.
strcspn ("hello, world", ",");
     => 5

       char *strpbrk(const char *s, const char *accept);
string point break라 읽으며 하는 읽은 strcspn과 비슷하다. 다른점은 accept에
정의된 문자가 처음 나오는 곳의 포인터를 넘겨준다는 점이다.
[예제]
strpbrk ("hello, world", " tn,.;!?")
     => ", world"

       char *strstr(const char *haystack, const char *needle);
strchr과 비슷한데 다른점은 한문자가 아닌 문자열을 찾는다는 것이다.
널문자는 비교하지 않는다.
리턴값은 찾으면 처음 발생한 곳의 포인터를 못찾으면 널을 리턴한다.
[예제]
strstr ("hello, world", "l")
     => "llo, world"
strstr ("hello, world", "wo")
     => "world"

       void *memmem(const void *haystack, size_t haystacklen,
                     const void *needle, size_t needlelen);
haystacklen 길이만큼의 haystack에서 needle의 needlelen만큼의 스트링을 찾아서
처음 발생한 번지를 리턴한다.
이함수는 GNU 확장이다. libc 5.0.9에서 이 함수를 사용하는 것은 위험하다. 거기에는
needle와 haystack 인자가 서로 바뀌어 있고 needle이 처음 발생한 곳의 끝 번지를
리턴한다. libc 5.0.9는 여전히 많이 사용되고 있기때문에 이 함수의 사용은 위험
하므로 사용을 자제해야 한다.
또한 libc나 glibc둘다 다음과 같은 버그가 있다.
만약 needle이 널이라면 리턴값은 haystack가 아니라 haystack - 1이 된다.
glibc 2.0.5는 더욱 나쁜 상황을 연출하는데 이는 haystack의 마지막 바이트의
포인터를 리턴한다. 따라서 결코 needle를 널로 주고 사용해서는 안되며
정 필요한 경우가 아니면 사용을 하지 말아야 할 함수이다.

       char *index(const char *s, int c);
strchr의 다른 이름으로 하는일은 똑같다

       char *rindex(const char *s, int c);
strrchr의 다른 이름으로 하는일은 똑같다.

              
2.5 토큰 함수
이들은 간단한 구분 분석이나 파싱을 할때 사용된다.
       char *strtok(char *s, const char *delim);
이 함수는 이해하는데 약간의 노력을 요하는 함수로 주의를 기울여야 할것이다.

char a[] = "하하하 이것은 토큰을 위한 예입지요";
char *p1, *p2, *p3, *p4, *p5;
p1 = strtok(a, " ");
p2 = strtok(NULL, " ");
p3 = strtok(NULL, " ");
p4 = strtok(NULL, " ");
p5 = strtok(NULL, " ");
delim이 처음 발생할 때까지의 문자를 찾고 그것을 널문자로 대체하고 문자열의
포인터를 리턴한다. 따라서 p1은 '하'를 가르키는 포인터가 된다. 그리고 다음
단어의 시작을 가르키는 포인터를 내부적으로 기억한다. 그렇기 때문에 첫번째
인자를 더이상 줄필요가 없으므로 NULL을 쓰는것이다.
다음 예제를 보면 확실히 이해가 될것이다.
char string[] = "words separated by spaces -- and, punctuation!";
const char delimiters[] = " .,;:!-";
char *token;

...

token = strtok (string, delimiters);  /* token => "words" */
token = strtok (NULL, delimiters);    /* token => "separated" */
token = strtok (NULL, delimiters);    /* token => "by" */
token = strtok (NULL, delimiters);    /* token => "spaces" */
token = strtok (NULL, delimiters);    /* token => "and" */
token = strtok (NULL, delimiters);    /* token => "punctuation" */
token = strtok (NULL, delimiters);    /* token => NULL */

주의해야 할점은 strtok를 사용하면 string 함수의 내용이 바뀌게된다.
따라서 const형 string에는 사용할 수 없다. 또한 내용이 유지되야 하는 스트링에
대해 사용하려면 먼저 복사를 해논다음에 사용해야 한다.

string의 내용을 한문자씩 출력 해보면 다음과 같다.
wordsseparatedbyspaces-- and punctuation
delim에 주어진 문자들은 널포인터로 바뀌기 때문에 띄어쓰기가 안된것처럼 보인다.
"-- " 부분은 앞의 공간이 널로 변환되고 delim에 주어지지 않은 문자의 시작인
a를 가르키기 때문에 그와 같이 출력되고 and다음의 공간은 and 바로 다음의 ','가
널문자로 치환되고 delim에 주어지지 안은 첫단어의 시작인 punc~로 가기때문에
그와 같이 출력되는 것이다.

char * strtok_r (char *newstring, const char *delimiters, char **save_ptr)
strtok와 비슷하다. 차이점은 strtok와 달리 다음 토큰의 시작을 내부적으로
기억하지 않는다는 것이다. 따라서 호출할때 save_ptr을 수동으로 제공해 줘야 한다.
2번째 호출부터 newstring을 널로 하고 save_ptr을 그대로 주면 strtok와 같이
작동하게 된다.
이 함수는 POSIX.1b에서 제안되었다.

       char *strsep(char **stringp, const char *delim);
#include <string.h>
#include <stddef.h>

...

char string[] = "words separated by spaces -- and, punctuation!";
const char delimiters[] = " .,;:!-";
char *running;
char *token;

...

running = string;
token = strsep (&running, delimiters);    /* token => "words" */
token = strsep (&running, delimiters);    /* token => "separated" */
token = strsep (&running, delimiters);    /* token => "by" */
token = strsep (&running, delimiters);    /* token => "spaces" */
token = strsep (&running, delimiters);    /* token => "and" */
token = strsep (&running, delimiters);    /* token => "punctuation" */
token = strsep (&running, delimiters);    /* token => NULL */

첫번째 인자를 부가적으로 줌으로써 중복적인 접근을 피할 수있다.
움직이는 포인터의 초기화는 사용자가 할수있다. strsep가 성공하면
delim에 의해 구분되는 토큰을 따라 포인터는 움직이게 된다.
리턴값은 다음 토큰의 처음을 가르키는 포인터이다. delim이 발견되지 않으면
null을 리턴한다.
4.3BSD에서 나온 함수이다.
2.6 기타 함수
       char *strerror(int errnum);
errnum에 넘겨진 에러코드를 설명하는 문자열을 리턴한다.
다음은 1 부터 19까지 errnum에 대한 설명이다.
errornum 0 : 성공
errornum 1 : 명령이 허용되지 않음
errornum 2 : 그런 파일이나 디렉토리가 없음
errornum 3 : 그런 프로세스가 없음
errornum 4 : 중단된 시스템 호출
errornum 5 : 입력/출력 오류
errornum 6 : 장치가 설정되지 않았음
errornum 7 : 인수 명단이 너무 깁니다
errornum 8 : Exec 형식 오류
errornum 9 : 잘못된 파일 기술자
errornum 10 : 자식 프로세스가 없음
errornum 11 : 자원이 일시적으로 사용 불가능함
errornum 12 : 메모리를 할당할 수 없습니다
errornum 13 : 허가 거부됨
errornum 14 : 잘못된 주소
errornum 15 : 블럭 장치가 필요함
errornum 16 : 장치나 자원이 동작 중
errornum 17 : 파일이 존재합니다
errornum 18 : 부적절한 장치간 연결
errornum 19 : 그런 장치가 없음

       char *strfry(char *string);
문자열을 rand()를 사용하여 랜덤화 하고 랜덤화된 문자열에 대한 포인터를
리턴한다.

[참고자료]
  Linux man page
  Programming Language - C
  love_C ( Cambridge University Engineering Department)
  The GNU C Library Reference Manual


'개발지식창고 > C' 카테고리의 다른 글

define함수 대신 template를 쓰자  (0) 2010.09.20
define 대신 inline을 쓰자  (0) 2010.09.20
BASE64 Encode/Decode  (0) 2010.09.20
문자열 처리 함수들  (0) 2010.08.22
printf 특수문자 출력하기  (0) 2010.08.16
Posted by 모과이IT
,


#include <STDIO.H>
#include <MALLOC.H>
#include <TCHAR.H>

static const char Base64_EM[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; //!< Base64 인코드 테이블
static   char Base64_DM[256] = {0,};                 //!< Base64 디코드 테이블


/*!
\brief  
 Base64 초기화 함수
 제공되는 다른 함수를 사용하기 이전에 선행 호출
\see  
*/
void base64_Init()
{
 for(int i = 0; i<256; i++)
  Base64_DM[i] = -1;

 for(int i = 0; i<64; i++)
  Base64_DM[Base64_EM[i]] = i;

 return;
}

/*!
\brief
 Base64 Encode 함수
 메모리의 내용을 문자열로 변환하여 주는 함수

\param[in] pInBuf    문자열로 바꿀 메모리의 포인터
\param[in] ulSize   문자열로 바꿀 메모리의 크기
\param[out] ppOutBuf   변환된 문자열을 받을 메모리 포인터이며, 내부에서 할당하여 반환하므로 사용이 끝나면 free 하여야 한다.

\return  할당된 ppOutBuf 메모리의 크기
\see  
*/
int base64_Encode(unsigned char *pInBuf, unsigned long ulSize, unsigned char **ppOutBuf)
{
 unsigned char input[3]  = {0,};
 unsigned char output[4] = {0,};

 unsigned long ulOutbufSize = 0;

 unsigned long ulOctaPosition = 0;
 unsigned long ulOctaSplit  = 0;
 unsigned long ulHexaSplit  = 0;

 unsigned char *pOctaBit  = 0;
 unsigned char *pOctaBit_endof = 0;

 pOctaBit_endof = pInBuf + ulSize - 1;
 ulOutbufSize = (4 * (ulSize / 3)) + (ulSize % 3? 4 : 0) + 1;

 (*ppOutBuf)  = (unsigned char*)malloc(ulOutbufSize);

 for  (pOctaBit = pInBuf;pOctaBit <= pOctaBit_endof; ulOctaSplit++, pOctaBit++) {
  ulOctaPosition   = ulOctaSplit % 3;
  input[ulOctaPosition] = *pOctaBit;

  if (ulOctaPosition == 2 || pOctaBit == pOctaBit_endof) {
   output[0] = ((input[0] & 0xFC) >> 2);
   output[1] = ((input[0] & 0x3) << 4) | ((input[1] & 0xF0) >> 4);
   output[2] = ((input[1] & 0xF) << 2) | ((input[2] & 0xC0) >> 6);
   output[3] = (input[2] & 0x3F);

   (*ppOutBuf)[ulHexaSplit++] = Base64_EM[output[0]];
   (*ppOutBuf)[ulHexaSplit++] = Base64_EM[output[1]];
   (*ppOutBuf)[ulHexaSplit++] = ulOctaPosition == 0? '_' : Base64_EM[output[2]];
   (*ppOutBuf)[ulHexaSplit++] = ulOctaPosition <  2? '_' : Base64_EM[output[3]];

   input[0] = input[1] = input[2] = 0;
  }
 }

 (*ppOutBuf)[ulHexaSplit] = '\0';

 return ulOutbufSize;
}

/*!
\brief
 Base64 Decode 함수
 문자열을 원본 데이터로 변환하여 주는 함수

\param[in] pInBuf    변환할 문자열
\param[in] ulSize   복원할 사이즈, Encode시의 Return 값과 같다.
\param[out] pOutBuf   복원 내용을 받을 메모리 포인터

\return  복원된 데이터의 사이즈
\see base64_Encode 
*/
int base64_Decode(unsigned char *pInBuf, unsigned long ulSize, unsigned char *pOutBuf)
{
 unsigned long ulOctaPosition = 0;
 unsigned long ulHexaPosition = 0;
 
 unsigned char* pOctaBit  = 0;
 unsigned char cOctaBit  = 0;
 char   cHexaBit  = 0;
 char   cHexaBit_prev = 0;

 for ( pOctaBit = pInBuf; *pOctaBit != '\0'; ++pOctaBit ) {
  cHexaBit = Base64_DM[(int) *pOctaBit];

  if ( cHexaBit != -1 ) {
   switch ( ulHexaPosition ) {
    case 0:
     ++ulHexaPosition;
     break;
    case 1:
     cOctaBit = ( ( cHexaBit_prev << 2 ) | ( ( cHexaBit & 0x30 ) >> 4 ) );
     if ( ulOctaPosition < ulSize )
      pOutBuf[ulOctaPosition++] = cOctaBit;
     ++ulHexaPosition;
     break;
    case 2:
     cOctaBit = ( ( ( cHexaBit_prev & 0xf ) << 4 ) | ( ( cHexaBit & 0x3c ) >> 2 ) );
     if ( ulOctaPosition < ulSize )
      pOutBuf[ulOctaPosition++] = cOctaBit;
     ++ulHexaPosition;
     break;
    case 3:
     cOctaBit = ( ( ( cHexaBit_prev & 0x03 ) << 6 ) | cHexaBit );
     if ( ulOctaPosition < ulSize )
      pOutBuf[ulOctaPosition++] = cOctaBit;
     ulHexaPosition = 0;
     break;
   }
   cHexaBit_prev = cHexaBit;
  }
 }

 return ulOctaPosition;

}

Posted by 모과이IT
,
** 문자열 처리 함수 **

#include

1) 문자열의 길이 측정(strlen)

strlen(문자열)

2) 문자열 결합(strcat)

char *ptr;
ptr = strcat(문자열, 문자열2)

3) 문자열 비교(strcmp)

int i;
i = strcmp(문자열, 문자열2)
i == 0 <--- 같다.
i != 1 같지 않다.

4) 문자열 복사(strcpy)

char *ptr;
strcpy(문자열, 문자열2)
문자열2를 문자열로 복사 5) 문자열 변환(atoi, atol, atof)

atoi -> 정수
atol -> long
atof -> 부동 소수점.

long b; float c;
int a;

a = atoi("1234" );
b = atol("-544334" );
c = atof("43.5456);

6) 문자열 교환 함수 (str_swap)

void str_cpy(char *ptr1, char *ptr2)
{
while(*ptr2)
*ptr1++ = *ptr2++;
*ptr1 = NULL;

}

void str_swap(char *a, char *b)
{
char temp[255];
str_cpy(temp, a)
str_cpy(a, b);
str_cpy(b, temp);
}
7) 문자열 내에서 특정 문자의 개수를 카운트 하는 함수(char_cnt)

int char_cnt(char *ptr, char ch)
{
int i = 0;

wile(*ptr)
if(*ptr++ == ch)
i++;

return(i);

}  

8) 문자열 내의 특정 문자를 다른 문자로 바꾸는 함수(str_chg)

int str_chg(char *ptr, char ch1, char ch2)
{
while(*ptr)
{
if(*ptr == ch1)
*ptr = ch2;

ptr++;
}
}

9) 문자열 내의 특정 문자를 탐색(str_fine)

char *str_find(char *ptr, char ch)
{
while(*ptr)
{
if(*ptr == ch)
return(ptr);
ptr++;
}

return((char *)(-1));

}

10) 문자열 내의 소문자를 대문자로 바꾸는 함수(str_upp)

void str_upp(char *ptr)
{
while(*ptr)
{
if(islower(*ptr))
*ptr = toupper(*ptr);
ptr++;
}
}

11) 문자열 내의 대문자를 소문자로 바꾸는 함수(str_low)

void str_low(char *ptr)
{
while(*ptr)
{
if(isupper(*ptr))
*ptr = tolower(*ptr);
ptr++;
}
}

12) 대문자와 소문자 상호 교환 함수(str_case)

void str_case(char *ptr)
{
while(*ptr)
{
if(isupper(*ptr))
*ptr = tolower(*ptr);
else if(islower(*ptr)
*ptr = toupper(*ptr);
ptr++;
}
}

13) 각 문장의 첫 문자만 대문자로 교환하는 함수(str_fst)

void str_fst(char *ptr)
{
if(islower(*ptr))
*ptr = toupper(*ptr);

while(*ptr)
{
if(*ptr == '.')
{
ptr++;

while(*ptr == ' ')
ptr++;

if(islower(*ptr))
*ptr = toupper(*ptr);
}

break;
}

ptr++;
}

14) 문자열 내의 지정한 위치에 다른 문자열을 삽입하는 함수(str_ins)

char *str_ins(char ptr1[], char ptr2[],int t)
{
char temp[255];
int i, j, k;

if(t>=str_len(ptr1) return (-1);

for(k=0; i=0; i<t; i++; k++)
temp[k] = ptr1[i];

for(j=0; ptr2[j] != NULL; j++, k++)
temp[k] = ptr2[j];

while(ptr1[i])
temp[k++] = ptr1[i++];
temp[k] = NULL;
}

15) 문자열 앞에 지정한 개수 만큼의 공백을 추가하는 함수(str_blk)

void str_blk(char ptr[], int t)
{
static char temp[255];
int i;

for(i = 0; i<t; i++)
temp[i] = BLANK;

temp[i] = NULL;

str_cat(temp, ptr);

str_cpy(ptr, temp);
}

16) 문자열 내의 모든 공백을 삭제하는 함수(rmv_blk)

void rmv_blk(char ptrp[])
{
char temp[255];
int i, k;

for(i=0, k=0; ptr[i] != NULL; i++)
if(ptr[i] != BLANK)
temp[k++] = ptr[i];

temp[k] = NULL;

str_cpy(ptr, temp);
}

17) 문자열 내에서 원하는 개수 만큼 문자를 삭제하는 함수(str_rmv)

void str_rmv(char *ptr, int loc, int count)
{
int len, i, j;
len = str_len(ptr);

if(loc >= len)
return (-1);

if(loc + count <= len)
{
j = loc + count;

for(i=loc; ptr[j] != 0; i++)
{
ptr[i] = ptr[j];
j++;
}
ptr[i] = NULL
}
else
ptr[loc] = NULL;
}

18) 특정한 문자열이 기억된 위치를 계산하는 함수(str_loc)

int str_loc(char ptr1[], char ptr2[])
{
int i, j, k;
for(i=0; ptr1[i] != NULL; i++)
for(j=i, k=0; ptr2[k] == ptr1[j]; k++, j++)
return (i);
return (-1);
}

19) 특정한 문자열이 나온 개수를 카운트 하는 함수(str_cnt)

int str_cnt(char ptr1[], char ptr2[])
{
int i, j, k;
int count = 0;
for(i=0; ptr1[i] != NULL; i++)
for(j=i, k=0; ptr2[k] == ptr1[j]; k++, j++)
if(ptr2[k+1] == NULL)
{
count++;

break;
}
return ((count == 0) ? -1 : count);
}

'개발지식창고 > C' 카테고리의 다른 글

C 문자열 함수 정리  (0) 2010.09.20
BASE64 Encode/Decode  (0) 2010.09.20
printf 특수문자 출력하기  (0) 2010.08.16
32비트 배열을 이용한 상태값 저장 및 얻어오기  (0) 2010.08.16
[C/C++]#define 매크로 팁  (0) 2010.08.16
Posted by 모과이IT
,

printf의 마지막 f formatted(서식화된)를 의미한다.

 

    다양한 종류의 특수 문자

특수 문자

 

\a

경고음 소리 발생

\b

백스페이스(backspace)

\n

개행(한줄띄고 처음으로(캐리지 리턴)하는것을 말한다)

\r

캐리지리턴

\t

수평 탭

\v

수직 탭

\\

백슬래시(\)

\’

작은 따옴표

\”

큰 따옴표

 

서식 문자의 종류와 의미

서식 문자

출력 형태

%c

단일 문자

%d

부호 있는 10진 정수(int)

%s

문자열

%o

부호 없는 8진 정수

%u

부호 없는 10진 정수

%x

부호 없는 16진 정수, 소문자 사용

%X

부호 없는 16진 정수, 대문자 사용

%e

e 표기법에 의한 실수

%E

E 표기법에 의한 실수

%g

값에 따라 %f , %e 둘 중 하나를 선택

%G

값에 따라 %F , %E 둘 중 하나를 선택

%%

% 기호 출력(ex:3% -> %%d)

 

필드 폭

   출력되는 필드의 폭을 지정 할 수 있다.

서식 문자

출력의 형태

%8d

필드 폭을 8칸 확보하고 오른쪽 정렬해서 출력

%-8d

필드 폭을 8칸 확보하고 왼쪽 정렬해서 출력

%+8d

필드 폭을 8칸 확보하고 오른쪽 정렬한 상태에서 양수는 +, 음수는 ?를 붙여서 출력

 

'개발지식창고 > C' 카테고리의 다른 글

C 문자열 함수 정리  (0) 2010.09.20
BASE64 Encode/Decode  (0) 2010.09.20
문자열 처리 함수들  (0) 2010.08.22
32비트 배열을 이용한 상태값 저장 및 얻어오기  (0) 2010.08.16
[C/C++]#define 매크로 팁  (0) 2010.08.16
Posted by 모과이IT
,


[헤더 파일에 선인]
ULONG32 statusDropDownControl; // 동적인 컨트롤 상태를 담기위한 변수 초기화값은 0x00000000
enum
 {
  DROPDOWN_NONE   = 0x00000000, // 0000 0000 0000 0000 0000 0000 0000 0000
  DROPDOWN_COLOR   = 0x00000001, // 0000 0000 0000 0000 0000 0000 0000 0001
  DROPDOWN_SIZE   = 0x00000002, // 0000 0000 0000 0000 0000 0000 0000 0010
  DROPDOWN_FRONT   = 0x00000004, // 0000 0000 0000 0000 0000 0000 0000 0100
  DROPDOWN_FRAMECOLOR  = 0x00000008, // 0000 0000 0000 0000 0000 0000 0000 1000

  DROPDOWN_UPDATE1   = 0x00000010, // 0000 0000 0000 0000 0000 0000 0001 0000
  DROPDOWN_UPDATE2   = 0x00000020, // 0000 0000 0000 0000 0000 0000 0010 0000
  DROPDOWN_UPDATE3   = 0x00000040, // 0000 0000 0000 0000 0000 0000 0100 0000
  DROPDOWN_UPDATE4   = 0x00000080, // 0000 0000 0000 0000 0000 0000 1000 0000
 }; 

[실제 cpp 구현부분]

 * 상대 값들을 추가 할때
 statusDropDownControl |= DROPDOWN_COLOR;
 statusDropDownControl |= DROPDOWN_SIZE;
 statusDropDownControl |= DROPDOWN_FRONT;
 statusDropDownControl |= DROPDOWN_FRAMECOLOR;

* 해당 상태값을 얻을때
if(statusDropDown & DROPDOWN_COLOR)
 {
  ..

  retCount ++;
 }
 
 if(statusDropDown & DROPDOWN_SIZE)
 {
  // Size
  if(SalePriceDlgFormWnd->bb.hStatic[retCount] == NULL)
  ...
  retCount ++;
 }

 if(statusDropDown & DROPDOWN_FRONT)
 {
  // FRONT
  if(SalePriceDlgFormWnd->bb.hStatic[retCount] == NULL)
  ..
  retCount ++;
 }
 
 if(statusDropDown & DROPDOWN_FRAMECOLOR)
 {
  // FRAMECOLOR
  if(SalePriceDlgFormWnd->bb.hStatic[retCount] == NULL)
 ..
  retCount ++;
 }

'개발지식창고 > C' 카테고리의 다른 글

C 문자열 함수 정리  (0) 2010.09.20
BASE64 Encode/Decode  (0) 2010.09.20
문자열 처리 함수들  (0) 2010.08.22
printf 특수문자 출력하기  (0) 2010.08.16
[C/C++]#define 매크로 팁  (0) 2010.08.16
Posted by 모과이IT
,

C/C++ 에서 #define 을 사용을 가급적 자제하도록 권고되고 있지만, 잘 사용할 경우 많은 코딩상의 이득을 볼 수 있습니다.  

#define을 활용하는 예를 몇 가지 설명하려고 합니다.

 

우선 팁을 설명하기 전에 간단하게 ‘##’ “#’ 에 대해 살펴보고 가겠습니다. (샘플소스는 MSDN 에서 가져왔습니다)

 

1.     ‘##’ - Token-Pasting Operator (##): 분리되어 있는 2개의 토큰을 하나로 뭉쳐주는 역할을 합니다.

#define paster( n ) printf( "token" #n " = %d", token##n )


라고 할 때 아래와 같은 코딩은

int token9 = 9;

paster( 9 ); // token##n -> token 과실제인수9 를합쳐token9 가됨


다음과 같이 풀이되어 결국 ‘token9 = 9’ 가 화면에 표시되게 됩니다.

printf( "token" "9" " = %d", token9 );

printf( "token9 = %d", token9 );


이와 같이 ## 은 왼쪽 Operand 와 오른쪽 Operand를 접착제처럼 붙여 주는 역할을 합니다.

 

2.     ‘#’ - Stringizing Operator (#): 매크로 인자를 문자열로 만드는 역할을 합니다.

#define stringer( x ) printf( #x "\n" )


과 같이 정의 할 경우 아래와 같은 문장은

stringer( In quotes in the printf function call\n );


다음과 같이 풀어쓰게 됩니다.

printf( "In quotes in the printf function call\n" "\n" );

 

이제 #define 매크로를 이용한 활용예를 들어 보겠습니다.

1.    배열의 크기 구하기


#define 을 이용하는 가장 간단하고 자주 쓰이는 팁이 아닐까 생각됩니다.

#define arrayCnt( x ) sizeof(x) / sizeof(x[0])


 이를 이용한 간단 샘플입니다.

struct LIST_COL_DEF

{

           int nChar;

           char szText[20];

} ListCol[] =

{

           {20, "폴더"},

           {20, "파일명"}

};

for (int i = 0; i < arrayCnt(ListCol); i++)

{

           // To Do Something...

}

 

물론

i < arrayCnt(ListCol)

를 사용하지 않고

i < sizeof(ListCol) / sizeof(ListCol[0])

을 사용해도 되지만 이와 같은 작업을 여러 번 반복할 경우(array 를 사용하는 경우는 정말 흔하디 흔하죠) 단조로운 코딩이 반복될 수 있어 #define 을 이용한 매크로를 애용하고 있습니다.

 

2.    Get/Set 함수 간단하게 만들기

OOP(Object Oriented Programming)로 프로그램 할 경우 멤버 변수의 Get 함수와 Set 함수를 만들어야 할 경우가 많습니다. 변수를 외부로 드러내지 않고 변수에 대한 접근 Method 만 제공함으로써 추후 변경에 대한 유지보수를 최소화하기 위해 많이 사용하게 되는데요(OOP 의 특징 중에 하나인 Encapsulation). 간단한 Get/Set 함수라 하더라도 하드코딩일 경우 손이 많이 가게 됩니다. 이럴 경우 #define 매크로를 사용하면 단순반복적인 Get/Set 함수를 한 줄만으로 코딩하여 작성할 수 있습니다.


예를 들어 다음과 같은 매크로를 정의한 경우

// SIMPLE_FUNC_IMPL -> Get/Set 함수를자동으로만들어줍니다.

// 1. ret -> return type

// 2. fname -> Get/Set 다음에올함수명

// 3. var -> Get/Set 에대상이되는변수명

#define SIMPLE_FUNC_IMPL(ret, fname, var) \

           ret Get##fname() \

           { \

                     return var; \

           } \

           void Set##fname(ret tmp) \

           { \

                     var = tmp; \

           }

 

bool m_bTest Get 하고 Set 하는 간단한 함수는 아래 한 줄로 만들 수 있게 됩니다.

SIMPLE_FUNC_IMPL(bool, Test, m_bTest);


위 한 줄의 매크로는 자동으로 전 처리기에 의해 컴파일 시 다음과 같은 함수로 확장됩니다.

bool GetTest()

{

           return m_bTest;

}

void SetTest(bool tmp)

{

           m_bTest = tmp;

}


간단하지 않은가요 ^^.

외부로 공개해야 하는 속성이 많으면 많을수록 코딩부담을 줄이고, 미스타이핑으로 인한 오류를 사전에 막아 줄 수 있어 매크로는 더욱 위력을 발휘하게 됩니다.

사실 마이크로소프트의 ATL 3.0 라이브로리 소스 헤더를 분석해 보면 이와 같은 매크로를 훨씬 복잡하게 사용하고 있다는 걸 알 수 있습니다.

아래 샘플은 ATL 3.0 라이브러리 ATLCTL.H 소스에 있는 코드입니다. (소스가 길어 접습니다

 

소스코드 닫기

 

#define IMPLEMENT_STOCKPROP(type, fname, pname, dispid) \

           HRESULT STDMETHODCALLTYPE put_##fname(type pname) \

           { \

                     __if_exists(T::m_##pname) \

                     { \

                                ATLTRACE(ATL::atlTraceControls,2,_T("CStockPropImpl::put_%s\n"), #fname); \

                                T* pT = (T*) this; \

                                if (pT->m_nFreezeEvents == 0 && pT->FireOnRequestEdit(dispid) == S_FALSE) \

                                          return S_FALSE; \

                                pT->m_##pname = pname; \

                                pT->m_bRequiresSave = TRUE; \

                                if (pT->m_nFreezeEvents == 0) \

                                          pT->FireOnChanged(dispid); \

                                __if_exists(T::On##fname##Changed) \

                                { \

                                          pT->On##fname##Changed(); \

                                } \

                                pT->FireViewChange(); \

                                pT->SendOnDataChange(NULL); \

                     } \

                     return S_OK; \

           } \

           HRESULT STDMETHODCALLTYPE get_##fname(type* p##pname) \

           { \

                     __if_exists(T::m_##pname) \

                     { \

                                ATLTRACE(ATL::atlTraceControls,2,_T("CStockPropImpl::get_%s\n"), #fname); \

                                ATLASSERT(p##pname != NULL); \

                                if (p##pname == NULL) \

                                          return E_POINTER; \

                                T* pT = (T*) this; \

                                *p##pname = pT->m_##pname; \

                     } \

                     return S_OK; \

           }

소스코드 닫기

 

 끝으로, 이와 같은 #define 매크로의 단점도 잠시 언급하면

-       디버깅이 힘듭니다. 디버깅 시 매크로의 경우 확장된 소스가 보여지지 않아 변수의 변경사항을 추적해 들어 가기가 힘듭니다. 매크로가 복잡할 경우 애로가 있을 수 있어 복잡한 함수의 경우는 위와 같은 방식을 취하지 않는 편이 좋습니다.

-       #define MAX_CUSTOMER 10 보다는
const int MAX_CUSTOMER = 10
으로 코딩 하는 것이 변수가 잘못 typecasting 되는 걸 막을 수 있는 좋은 코딩 습관입니다. 사실 이 부분 때문에 #define 매크로사용을 자제해야 한다고 권장한다고 봐야겠죠.

 

이상으로 #define 매크로에 대해 아주 간단한 팁을 알아봤습니다. 사실 초보자가 아닌 경우 대부분 아실 만한 내용이라 적기가 부끄러운 내용이긴 한데 기록해 둘 경우 나중에 기억이 가물가물해 졌을 때나, 프로그램을 시작하는 다른 분들에게 도움이 될까 하고 적어봤습니다. 오늘도 열코!!

'개발지식창고 > C' 카테고리의 다른 글

C 문자열 함수 정리  (0) 2010.09.20
BASE64 Encode/Decode  (0) 2010.09.20
문자열 처리 함수들  (0) 2010.08.22
printf 특수문자 출력하기  (0) 2010.08.16
32비트 배열을 이용한 상태값 저장 및 얻어오기  (0) 2010.08.16
Posted by 모과이IT
,