2010년 10월 22일 금요일

template 정리

C++은 여러 가지 개발 방법을 지원하는 멀티 패러다임 언어라고 하는데 적어도 다음 세 가지 방법으로 개발을 할 수 있다.

 

① 구조적 프로그래밍 : C언어에서와 마찬가지로 함수 위주로 프로그램을 작성할 수 있다. C++이 C언어의 계승자이므로 C언어의 개발 방법을 지원하는 것은 당연하다.

 

② 객체 지향 프로그래밍 : 캡슐화, 추상화를 통해 현실 세계의 사물을 모델링할 수 있으며 상속과 다형성을 지원하기 위한 여러 가지 언어적 장치를 제공한다.

 

③ 일반화 프로그래밍 : 임의 타입에 대해 동작하는 함수나 클래스를 작성할 수 있다. 객체 지향보다 재사용성과 편의성이 더 우수하다.

 

템플릿(Template)이란 무엇인가를 만들기 위한 형틀이라는 뜻이다.

 

1. 함수 템플릿

 

사용방법

template <typename T>

T Function(T a, T b)

{

   T..;

   .....;

}

 

사용예제 1.

template <typename T>

void func(void)

{

     T v;

 

     cin >> v;

     cout << v;

}

 

func<int>(); // 이런식으로 사용함.

 

사용예제 2.

template <typename T>

T cast(int s)

{

     return (T)s;

}

 

unsigned i = cast<unsigned>(1234);

 

함수의 호출부를 보고 컴파일러가 템플릿 함수를 알아서 만드는 것을 암시적 구체화라고 한다.

명시적 구체화

만약 특정 타입에 대한 템플릿 함수를 강제로 만들고 싶다면 이때는 명시적 구체화(Explicit Instantiation)를 하는데 이는 지정한 타입에 대해 함수를 생성하도록 컴파일러에게 지시하는 것이다. 예를 들어 float 타입을 교환하는 함수를 생성하고 싶다면 다음 명령을 사용한다.

 

template void Swap<float>(float, float);

 

2.클래스 템플릿

 

클래스 템플릿은 함수 템플릿과 비슷하되 찍어내는 대상이 클래스라는 것만 다르다.

ex)

class PosValueInt

{

private:

     int x,y;

     int value;

public:

     PosValue(int ax, int ay, int av) : x(ax),y(ay),value(av) { }

     void OutValue();

};

 

class PosValueChar

{

private:

     int x,y;

     char value;

public:

     PosValue(int ax, int ay, char av) : x(ax),y(ay),value(av) { }

     void OutValue();

};

 

class PosValueDouble

{

private:

     int x,y;

     double value;

public:

     PosValue(int ax, int ay, double av) : x(ax),y(ay),value(av) { }

     void OutValue();

};

 

위의 소스를 클래스 탬플릿 형태로 바꾸어 본다.

include <Turboc.h>

#include <iostream>

using namespace std;

 

template <typename T>

class PosValue

{

private:

     int x,y;

     T value;

public:

     PosValue(int ax, int ay, T av) : x(ax),y(ay),value(av) { }

     void OutValue();

};

 

template <typename T>

void PosValue<T>::OutValue()

{

     gotoxy(x,y);

     cout << value << endl;

}

 

void main()

{

     PosValue<int> iv(1,1,2);

     PosValue<char> cv(5,1,'C');

     PosValue<double> dv(30,2,3.14);

     iv.OutValue();

     cv.OutValue();

     dv.OutValue();

}

 

함수에서와 마찬가지로 클래스 템플릿도 단순한 선언에 불과하며 컴파일러는 이 템플릿의 모양을 기억해 두었다가 객체가 생성될 때 전달된 타입에 맞는 클래스 정의를 구체화한다.

템플릿으로부터 만들어지는 클래스도 분명히 클래스이며 일반적인 클래스와 전혀 다를 바가 없다. 템플릿 클래스로부터 상속하는 것도 가능하며 문법도 동일하되 기반 클래스의 이름에 < > 괄호가 사용되는 차이밖에 없다. 다음 클래스는 PosValue<int>로부터 새로운 클래스를 파생한다.

 

class PosValue2 : public PosValue<int> { ... }

 

템플릿 클래스가 다른 클래스의 기반 클래스로 사용되면 컴파일러는 클래스를 즉시 구체화한다. 설사 이 클래스의 인스턴스 선언문이 없더라도 말이다. 템플릿으로부터 만들어지지 않은 일반 클래스의 특정 멤버 함수만 템플릿으로 선언하는 것도 가능하다. 멤버 함수도 분명히 함수이므로 타입에 따라 여러 벌이 필요하다면 원하는 함수 하나만 함수 템플릿으로 만들면 된다. 다음 예제는 그 예를 보여 준다.

class Some

{

private:

     int mem;

 

public:

     Some(int m) : mem(m) { }

     template <typename T>

     void memfunc(T a) {

          cout << "템플릿 인수 = " << a << ", mem = " << mem << endl;

     }

};

void main()

{

     Some s(9999);

 

     s.memfunc(1234);

     s.memfunc(1.2345);

     s.memfunc("string");

}

비타입 인수

 

#include <Turboc.h>

 

template <typename T, int N>

class Array

{

private:

     T ar[N];

public:

     void SetAt(int n,T v) { if (n < N && n >=0) ar[n]=v; }

     T GetAt(int n) { return (n < N && n >=0 ? ar[n]:0); }

};

 

void main()

{

     Array<int,5> ari;

     ari.SetAt(1,1234);

     ari.SetAt(1000,5678);

     printf("%d\n",ari.GetAt(1));

     printf("%d\n",ari.GetAt(5));

}

]

Array<int,5> ari;

Array<int,5> ari2;

Array<int,6> ari3;

 

ari=ari2;

ari=ari3;        // 에러

 

클래스 선언문의 비타입 인수는 반드시 상수여야 하며 실행중에 값이 결정되는 변수는 인수로 사용할 수 없다. 다음 선언문은 에러로 처리된다.

 

int size=5;

Array<int,size> ari;

 

#include <Turboc.h>

 

template <int N>

void func(void)

{

     int ar[N];

 

     printf("배열 크기=%d\n",N);

}

 

void main()

{

     func<5>();

     func<8>();

}

 

func 템플릿은 비타입 인수 N을 요구하므로 func()라고만 호출해서는 지역 배열의 크기를 결정할 수 없으므로 함수를 구체화할 수 없다. 반드시 명시적인 템플릿 인수를 전달해야 한다. 이 예제는 gcc, 비주얼 C++ 7.0이상의 최신 컴파일러에서는 잘 실행되지만 비주얼 C++ 6.0에서는 컴파일되지 않는다. 비주얼 C++ 6.0은 클래스의 비타입 인수는 지원하지만 함수의 비타입 인수는 아직 지원하지 못한다.

 

디폴트 템플릿 인수

template <typename T=int>

class PosValue

{

     ....

 }

PosValue<> iv(1,1,2);

 

클래스 템플릿에는 디폴트 인수를 줄 수 있지만 함수 템플릿에는 디폴트를 정의할 수 없다. 클래스는 객체를 선언할 때 클래스 타입을 지정하므로 생략 가능하지만 함수는 호출할 때 실인수의 타입을 보고 구체화할 함수를 결정한다. 실인수가 생략되어 버리면 도대체 어떤 타입의 함수를 원하는지 컴파일러가 알 방법이 없기 때문이다.

 

특수화

클래스 템플릿도 함수 템플릿과 마찬가지로 실제 클래스 타입이 사용될 때만 구체화된다. 만약 특정 타입에 대해 미리 클래스 선언을 만들어 놓을 필요가 있다면 명시적 구체화를 할 수 있다. 예를 들어 float 타입의 PosValue 클래스를 미리 정의해 두고 싶다면 다음과 같이 한다.

 

template class PosValue<float>;

#include <Turboc.h>

#include <iostream>

using namespace std;

 

template <typename T>

class PosValue

{

private:

     int x,y;

     T value;

public:

     PosValue(int ax, int ay, T av) : x(ax),y(ay),value(av) { }

     void OutValue();

};

 

template <typename T>

void PosValue<T>::OutValue()

{

     gotoxy(x,y);

     cout << value << endl;

}

 

struct tag_Friend {

     char Name[10];

     int Age;

     double Height;

};

 

template <> class PosValue<tag_Friend>

{

private:

     int x,y;

     tag_Friend value;

public:

     PosValue(int ax, int ay, tag_Friend av) : x(ax),y(ay),value(av) { }

     void OutValue();

};

 

void PosValue<tag_Friend>::OutValue()

{

     gotoxy(x,y);

     cout << "이름:" << value.Name << ", 나이:" << value.Age

          << ", 키:" << value.Height << endl;

}

 

void main()

{

     PosValue<int> iv(1,1,2);

     tag_Friend F={"아무개",25,177.7};

     PosValue<tag_Friend> fv(2,2,F);

     iv.OutValue();

     fv.OutValue();

}

 

template<> class 클래스명<특수타입>

 

이렇게 정의하면 지정한 타입에 대해 특수화된 클래스를 생성한다. 인수의 타입이 이미 결정되어 있으므로 특수화된 클래스의 멤버 함수를 외부에서 정의할 때는 template < >를 붙이지 않아도 상관없다. OutValue 함수는 tag_Friend 구조체의 각 멤버를 순서대로 출력하도록 수정했는데 원래의 PosValue 템플릿에 있는 OutValue와는 코드가 다르다. 실행해 보면 (2,2) 위치에 구조체 F의 내용이 출력될 것이다.

특수화를 하면 특수화된 클래스는 객체를 선언하지 않더라도 자동으로 구체화된다. 즉, 클래스 정의가 만들어지고 멤버 함수들은 컴파일되어 실행 파일에 포함된다. 따라서 특수화된 클래스에 대한 정의는 일반적인 템플릿 클래스와는 달리 헤더 파일에 작성해서는 안되며 구현 파일에 작성해야 한다. 예제에서는 구조체에 대해서도 PosValue 템플릿을 쓰기 위해 특수화를 사용했는데 사실 이보다 더 간단한 방법은 tag_Friend 구조체가 << 연산자를 오버로딩해서 기존 템플릿의 본체 코드를 지원하는 것이다.

부분 특수화(Partial Specialization)란 템플릿 인수가 여러 개 있을 때 그 중 하나에 대해서만 특수화를 하는 기법이다. 다음 템플릿을 보자.

 

template <typename T1, typename T2> class SomeClass { ... }

 

SomeClass 클래스 템플릿은 두 개의 인수를 가지므로 <int, int>, <int, double>, <short, unsigned> 등 두 타입의 조합을 마음대로 선택할 수 있다. 부분 특수화는 이 중 하나의 타입은 마음대로 선택하도록 그대로 두고 나머지 하나에 대해서만 타입을 강제로 지정하는 것이다. T2가 double인 경우에 대해서만 특수화를 하고 싶다면 다음과 같이 한다.

 

template <typename T1> class SomeClass<T1, double> { ... }

 

이 상태에서 SomeClass<int, unsigned>나 SomeClass<float, short>는 특수화되지 않은 버전의 템플릿으로부터 생성되지만 SomeClass<int, double>이나 SomeClass<char, double>은 부분 특수화된 템플릿으로부터 생성될 것이다. 두 번째 인수가 double인 클래스에 대해서만 부분적으로 특수화를 했기 때문이다. gcc는 부분 특수화를 지원하지만 비주얼 C++ 6.0에서는 지원되지 않는다.

 

- 출처 . winapi 연구싸이트 -

 

====================================================================================

함수 템플릿

template <템플릿 매개 변수 목록>

반환형 함수이름

{

 

}

ex) template < typename T>

     T Plus( T a, T b)

     {

        return a + b;

     }

 

int a = Plus<int> (2,5); // ok

int b = Plus(2,5); //ok

float d = Plus(7,2.4f); // error 컴파일러가 어떤형을 T 에 대입 할줄 모른다.

 

함수 템플릿을 호출하는 구문은 컴파일러에 의해 템플릿 매개 변수로 넘긴 자료형으로 구성된 함수 코드로 컴파일 된다. 템플릿은 실제 실행되지 않지만 실행되는 코드를 만들기 위한 거푸집 역활을 한다.

템플릿에 의해 만들어 지는 실제 코드를 템플릿의 인스턴스 라고 한다.

함수 템플릿은 인스턴스 함수가 되고 클래스 템플릿은 인스턴스 클래스가 된다.

따라서 Plus<int> 는 Plus 함수 템플릿의 매겨 변수 T 대신 int 로 치환된 함수가 되고

Plus<float> 은 T 대신 Float 으로 치완된 함수로 생성된다.

 

template < typename T>

T Plus(T a, T b)

{

   return a + b;

}

 

int Plus<int> (3,5);

=>

int Plus(int a,int b)

{

    return a + b;

}

 

int a = Plus<float> (2.7f,5.0f)

 

float Plus(float a, float b)

{

   return a+ b;

}

댓글 없음:

댓글 쓰기