프로그래밍/typescsript

타입스크립트의 Generic(제네릭)

한코코 2022. 11. 23. 11:56

Generic(제네릭)이란?

C#이나 JAVA 등 언어에서 제네릭은 재사용성이 높은 컴포넌트를 만들때 자주 활용되는 특징이며, 한가지 타입보다 여러가지 타입에서 동작하는 컴포넌트를 생성하는데 사용된다. 타입스크립트에서 제네릭이란 타입을 마치 함수의 파라미터처럼 사용하는 것을 의미한다.

// 어떠한 타입이라도 넣을 수 있고, 그와 같은 타입으로 return 값을 지정하는 코드
function getText<T>(text: T): T { return text;}

getText<string>('hi');
getText<number>(10);
getText<boolean>(true);

제네릭을 위 코드처럼 사용함으로서 어떤 타입이던지 복잡하게 |을 쓰지않고도 사용할 수 있다.

 

 

 

제네릭 타입 변수

우선 T라는 변수 타입을 받고, 인자값으로는 배열형태의 T를 받는 코드다.

예) [1,2,3]같은 배열을 받으면 반환값으로 number를 돌려줌.

function logText<T>(text: T[]): T[] {
  console.log(text.length); // 제네릭 타입이 배열이기 때문에 `length`를 허용합니다.
  return text;
}

// 같은 내용이지만 좀더 명시적으로 작성한 코드
function logText<T>(text: Array<T>): Array<T> {
  console.log(text.length);
  return text;
}

 

 

 

제네릭 타입 간략화

다음 #1과 #2는 같은 기능을 하는 코드다. #2가 좀더 단순화되어있다.

function logText<T>(text: T): T { return text; }

let str: <T>(text: T) => T = logText; // #1
let str: {<T>(text: T): T} = logText; // #2

이처럼 제네릭 인터페이스 코드를 다음과 같이 작성할 수 있다.

function logText<T>(text: T): T { return text; }

interface GenericLogTextFn { <T>(text: T): T; }
let myString: GenericLogTextFn = logText;

interface GenericLogTextFn<T> { (text: T): T; } // 인자타입 강조
let myString: GenericLogTextFn<string> = logText;

 

 

 

제네릭 클래스

제네릭 인터페이스와 비슷하게 사용하면 되는 제네릭 클래스.

단, 제네릭 클래스는 정적 측면이 아닌 인스턴스 측면에서만 제네릭이므로 클래스로 작업할때 정적 멤버는 클래스의 유형 매개변수를 사용할 수 없다.

class GenericMath<T> {
  pi: T;
  sum: (x: T, y: T) => T;
}

let math = new GenericMath<number>();

 

 

 

제네릭 제약 조건

앞에서 제네릭 타입 변수에서 살펴본 내용 말고도 제네릭 함수에 어느 정도 타입 힌트를 줄 수 있는 방법이 있다.

해당 타입을 정의하지 않고도 length 속성 정도는 허용하려면 아래와 같이 작성한다.

interface LengthWise {
  length: number;
}

function logText<T extends LengthWise>(text: T): T {
  console.log(text.length);
  return text;
}

// Error, 숫자 타입에는 `length`가 존재하지 않으므로 오류 발생
logText(10);

// `text.length` 코드는 객체의 속성 접근과 같이 동작하므로 오류 없음
logText({ length: 0, value: 'hi' });

 

 

객체의 속성을 제약하는 방법

두 객체를 비교할 때도 제네릭 제약 조건을 사용할 수 있다.

제네릭을 선언할 때 <O extends keyof T> 부분에서 첫 번째 인자로 받는 객체에 없는 속성들은 접근할 수 없게끔 제한한 코드

function getProperty<T, O extends keyof T>(obj: T, key: O) {
  return obj[key];  
}
let obj = { a: 1, b: 2, c: 3 };

getProperty(obj, "a"); // okay
getProperty(obj, "z"); // error: "z"는 "a", "b", "c" 속성에 해당하지 않습니다.

 

 

참고 사이트 : 타입스크립트 핸드북