참고한 글과 강의
노마드 코더 타입스크립트로 블록체인 만들기
타입스크립트 핸드북(joshua1988) 타입추론
타입스크립트 도큐멘테이션
기록의 힘 [TypeScript] 타입스크립트 함수 오버로딩 : Function Overloading
토스트 UI 타입스크립트의 Never 타입 완벽 가이드
Types vs. interfaces in TypeScript
Type vs Interface, 언제 어떻게?

타입스크립트를 쓰는 이유

자바스크립트는 매우 친절한 언어다. 그래서 프로그래머가 어떤 실수를 했던지 간에 일단 실행을 시키고 본다. 그러다 보니 자바스크립트로 설계를 할 때, 프로그래머가 원하는 타입을 얻지 못하는 경우가 있는데 ‘휴먼 에러'도 여기에 속한다. 타입스크립트는 강타입 언어이기 때문에 타입을 강제한다. 함수를 설계하든 변수를 만들든 무엇을 만들던지 간에 정확하게 내가 만드려고 하는 것의 타입이 무엇인지 반드시 알려줘야한다. 따라서 사람의 실수를 줄여줄 수 있으며 설계한대로 원하는 데이터를 얻을 수 있다.

타입스크립트는 많은 사람이 함께 코드를 작성하는 환경에 매우 적합하다. 코드를 나만 보고 나만 수정한다면 사람의 실수가 그다지 치명적이지 않을 수 있다. 하지만 대부분의 서비스는 많은 사람들이 함께 만들게되고 내가 작성한 코드는 다른사람이 더 많이 보고 수정하게 된다. 따라서 타입스크립트를 사용하게 되면 더 많은 정보를 다른 프로그래머에게 알려줄 수 있고 그만큼 유지 보수 측면에서 좋다. 서비스의 규모가 클수록 코드 관리를 조금 더 효율적으로 할 수 있다.

VSCode에서 Typescript를 사용하면 자동 완성 기능을 제공한다. 개발자가 타입을 정해놓으면 편집기가 알아서 정의한 타입을 사용할 수 있도록 코드 완성을 해준다. 또한 없는 타입이나 꼭 들어가야하는 값인데 누락되면 에러 메시지를 보여주기 때문에 예측 가능한(실행되기 전에 에러를 잡는) 프로그래밍을 할 수 있도록 도와준다. 또한 외부 패키지를 쓸 때, 패키지에 포함된 함수나 값에 어떤 값이 어떻게 들어가야하는지도 빠르게 탐색할 수 있다. 오류가 발생했을 때 왜 발생했는지 알 수 있고 어떻게 해결하는지 찾을 가능성이 높아진다.

Basic Usage

1// 원시 타입
2const age:number = 1
3// array
4const array:string[] = ['1', '2']
5// any
6const arrayAny:any = [1, '2', {}, [1,2]]
7// void
8const voidFunction = ():void => console.log('hi')
9// function
10const addUser = (name:string, age:number):{name:string, age:string} => ({name:"현수", age:32})
11// object
12const object :{name:string, age:number} ={
13 name:"안녕",
14 age:23
15}
16// optional
17const optionalFunc = (user:{name:string, age?:number}) =>{
18 console.log(name)
19}
20optionalFunc({name:"타노스"})
21optionalFunc({name:"스탠", age:96})
22// union
23function welcomePeople(x: string[] | string) {
24 if (Array.isArray(x)) {
25 // 여기에서 'x'는 'string[]' 타입입니다
26 console.log("Hello, " + x.join(" and "));
27 } else {
28 // 여기에서 'x'는 'string' 타입입니다
29 console.log("Welcome lone traveler " + x);
30 }
31}
32const unionArray: (string | number)[] = [1,2,3, "string"]
33const beNull : null = null
34const beUndefined = : undefined = undefined
35// tuple
36const tuple:[string, number] = ["스파이더맨", 2]

타입스크립트에서 타입을 정의하는 방법은 매우 다양하다. 기본적으로 변수 뒤에 :를 쓰고 내가 정의하고자 하는 타입을 붙일 수 있다. 타입은 원시 타입(string, number, boolean), any, array, object, tuple, enum, function, union, void, null, undefined, never, unknown이 있다.

타입 추론

타입 추론은 타입 스크립트가 코드를 해석해 나가는 동작을 의미한다.

1const name = "스트레인지";
2typeof name === string; // true

name은 string이라는 타입을 명시적으로 지정하지 않았지만 name은 string이다. 변수, 속성, 함수 반환 값 등을 설정할 때 타입 추론이 일어난다.

최적 공통 타입

여러 표현 식에서 타입을 추론할 때, 표현식들의 타입을 이용해 최적 공통 타입을 계산한다.

1let x = [0, 1, null];
2// x : (number | null)[]

이런식으로 타입이 추론되기를 원하지만 가끔 그렇지 않을 때가 있다. 그래서 명시적으로 타입이 무엇인지 알려주어야할 때도 있다.

문맥상 타이핑

표현식의 타입 위치에 의해 암시될 때 발생한다.

1window.onmousedown = function (mouseEvent) {
2 console.log(mouseEvent.button); //<- OK
3 console.log(mouseEvent.kangaroo); //<- Error!
4};

window.onmousedown 함수의 타입을 사용하여 오른쪽 함수 표현식의 타입을 추론했다. mouseEvent에 kangaroo가 없기 때문에 에러가 발생한다.

문맥상 타이핑은 함수 호출에 대한 인수, 오른쪽에 할당된 것, 타입 어셜션, 개체 및 배열 리터럴의 멤버, 반환문이 포함된다. 컨텍스트 타입은 가장 일반적인 타입에서 후보 타입으로 작동한다.

Type Aliases

1type User = {
2 name: string;
3 age: number;
4 isHuman?: boolean;
5};
6
7const captin: User = {
8 name: "캡틴",
9 age: 100,
10};
11
12const thor: User = {
13 name: "토르",
14 age: 5000,
15 isHuman: false,
16};
17
18const newUser = (info: User) => {
19 const { name, age, isHuman } = info;
20 return {
21 name,
22 age,
23 isHuman,
24 };
25};

type은 객체의 형태를 정할 때 사용한다. User라는 type을 만들면 여러 변수에 모양을 지정할 수 있다.

Call Signatures

1type MathFunc = (a: number, b: number) => number;
2
3const add: MathFunc = (a, b) => a + b;

Call signatures는 함수 타입을 지정할 때 사용한다. Call signatures를 지정하고 난 뒤에 작성한 함수에 적용하면 가독성이 높은 코드를 작성할 수 있다.

Overloading

1type MathFunc = {
2 (a: number, b: number): void;
3 (a: number, b: number): number;
4};
5
6const multi: MathFunc = (a, b) => {
7 const error = "숫자가 너무 큽니다.";
8
9 if (a * b >= a * a) {
10 console.log(error);
11 }
12
13 return a * b;
14};
15
16type Divided = {
17 (a: number, b: number): number;
18 (a: number, b: number, c: number): number;
19};
20
21const dividedFunc: Divided = (a, b, c?: number) => {
22 if (c) return (a / b) * c;
23 return a / b;
24};

Overloading은 함수가 여러개의 Call Signature를 가질때 발생한다.

Polymorphism & Generics

Polymorphism(다형성)은 용어가 가진 뜻 그대로 다양한 형태를 뜻한다. 함수가 다형성을 지닌다는 것은 내가 설계하려는 함수의 타입이 지정되어 있는 것이 아니라 예측 할 수 없는 다양한 형태를 수용할 수 있다는 뜻과 같다.

1// 다양한 형태를 수요할 수 없다.
2type PolyArray = {
3 (arr: number[]): void;
4 (arr: string[]): void;
5 (arr: (number | string)[]): void;
6 (arr: boolean[]): void;
7};
8
9const polyArrayPrinter: PolyArray = arr => {
10 arr.forEach(value => console.log(value));
11};
12
13polyArrayPrinter([1, 2, 3, 4]);
14polyArrayPrinter([1, 2, true, 4]); //error

다형성을 지닌 함수를 설계하기 위해서 제네릭 타입을 지정할 수 있다. 제네릭은 Call Signature를 프로그래머가 직접 지정하지 않아도 타입스크립트가 알아서 Call Signature를 생성해준다. 그래서 다양한 형태의 자료를 받을 수 있다.

1// 다양한 형태를 수요할 수 있다.
2type PolyArray = {
3 <T>(arr: T[]): void;
4 <T>(arr: T[]): T;
5};
6
7const polyArrayPrinter: PolyArray = arr => {
8 if (arr.length > 2) return arr[0];
9 return arr.forEach(value => console.log(value));
10};
11
12polyArrayPrinter([1, 2, 3, 4]);
13polyArrayPrinter(["1", "2", "3", "4"]);
14polyArrayPrinter([1, 2, true, 4]);
15
16// 이렇게도 작성할 수 있다.
17function polyArray<T>(arr: T[]) {
18 return arr[0];
19}
20
21// 화살표 함수는 이렇게 해주어야한다?
22// https://developer-talk.tistory.com/195
23const polyArray2 = <T extends {}>(arr: T[]) => arr[0];

any를 사용하지 않고 제네릭 타입을 사용하는 이유는 any는 타입체크를 하지 않지만 제네릭은 타입 체크를 하기 떄문이다.

Classes

1class User {
2 constructor(
3 private firstName: string,
4 private lastName: string,
5 public email: string
6 ) {}
7}
8
9const soo = new User("soo", "kang", "kangsoo@google.com");
10
11console.log(soo.email);
12console.log(soo.firstName); //error

객체 지향 프로그래밍을 할 때 자바스크립트에서 class를 사용한다. Typescript에서 class를 사용하여 위와 같이 구성할 수 있다.

abstract

1abstract class User {
2 constructor(
3 public firstName: string,
4 private lastName: string,
5 protected email: string
6 ) {}
7 abstract getEamil(): void;
8 getFullName() {
9 return `${this.firstName} ${this.lastName}`;
10 }
11}
12
13class Player extends User {
14 getEamil() {
15 console.log(this.email);
16 }
17}
18
19const hyun = new Player("hyun", "san", "sanhyun@naver.com");
20hyun.getFullName();
21const soo = new User("soo", "kang", "kangsoo@google.com"); // error

abstract는 상속만 받을 수 있는 class다. 인스턴스를 직접 생성할 수 없다. 클래스에서 public, private, protected 속성을 사용할 수 있다. public은 모든 곳에서 읽기와 쓰기가 가능하다. private는 정의된 클래스 안에서만 사용 가능하고 상속받은 곳에서도 사용할 수 없다. protected는 private와 같은 역할을 하지만 상속받은 클래스에서는 사용이 가능하다.

메소드도 abstract이 가능하다. 메소드를 추상화하는 것은 call signature를 만드는 것과 같다. 상속을 받는 자녀에서는 추상화 된 메소드를 구현해야한다.

1type Words = {
2 [key: string]: string;
3};
4
5class Dict {
6 private words: Words;
7 constructor() {
8 this.words = {};
9 }
10 add(word: Word) {
11 if (this.words[word.term] === undefined) {
12 this.words[word.term] = word.desc;
13 }
14 }
15 desc(term: string) {
16 return this.words[term];
17 }
18 deleteWord(term: string) {
19 delete this.words[term];
20 }
21}
22
23class Word {
24 constructor(public term: string, public desc: string) {}
25}
26
27const pizza = new Word("피자", "이탈리아 부침개");
28const dict = new Dict();
29
30dict.add(pizza);
31dict.desc("pizza");

출처 : 노마드 코더 타입스크립트 강의

Interface

인터페이스는 오브젝트의 모양을 특정해주는 용도로 사용할 수 있다.

1interface User {
2 name: string;
3 age: number;
4 money: number;
5}

interface에서 제네릭을 사용할 수 있다.

1interface OwnStorage<T> {
2 [key: string]: T;
3}
4
5class LocalStorage<T> {
6 private storage: OwnStorage<T> = {};
7 get(key: string): T {
8 return this.storage[key];
9 }
10 set(key: string, value: T) {
11 this.storage[key] = value;
12 }
13 remove(key: string) {
14 delete this.storage[key];
15 }
16 clear() {
17 this.storage = {};
18 }
19}
20
21const stringsStorage = new LocalStorage<string>();
22stringsStorage.get("user");
23stringsStorage.set("earth", "hi earth");
24
25const booleanStorage = new LocalStorage<boolean>();
26booleanStorage.get("true");
27booleanStorage.set("hello", true);

Types와 Interface의 차이점

type과 interface는 둘 다 오브젝트의 모양을 설명하는 용도로 사용이 가능하다. 하지만 type은 조금 더 다양한 목적으로 활용이 가능하지만 interface는 오직 오브젝트의 모양을 설명하는 목적으로만 사용이 가능하다.

1interface User {
2 name: string;
3}
4
5interface Player extends User {}
6
7type User = {
8 name: string;
9};
10
11type Player = User & {};
12
13const person1: Player = {
14 name: "unknown1",
15};

타입은 축척이 불가능하지만 인터페이스는 가능하다.(선언 병합이 가능하다.)

1interface User {
2 name: string;
3}
4interface User {
5 age: number;
6}
7
8type User = {
9 name: string;
10};
11// 불가능
12type User = {
13 age: number;
14};

인터페이스를 사용하면 추상 class를 사용하지 않으면서 class 형태를 지정할 수 있다. 추상 클래스는 자바스크립트로 컴파일이 되지만 interface는 컴파일이 되지 않기 때문에 조금더 가벼운 자바스크립트를 만들 수 있다.

1interface User {
2 firstName: string;
3 lastName: string;
4 email: string;
5 getEamil(): void;
6 getFullName(): string;
7}
8
9interface Auth {
10 auth: string;
11}
12
13class Player implements User, Auth {
14 constructor(
15 public firstName: string,
16 public lastName: string,
17 public email: string,
18 public auth: string
19 ) {}
20 getEamil() {
21 console.log(this.email);
22 }
23 getFullName() {
24 return `${this.firstName} ${this.lastName}`;
25 }
26}
27
28const hyun = new Player("hyun", "san", "sanhyun@naver.com", "lev.1");

인터페이스에서 타입을 상속하는 것은 가능할까?

해보니까 가능하다.

1type Human = {
2 name: string;
3};
4
5interface User extends Human {
6 age: number;
7}
8
9type Planet = "Earth" | "Moon" | "Mars";
10
11interface User {
12 planet: Planet;
13}

유니온 타입이 교차로 가능하다.

1interface Hunam {
2 name: string;
3}
4
5interface Alien {
6 name: string;
7}
8
9// type을 interface로 union하는 것은 불가능
10type Universe = Human | Alien;

언제 써야할까?

무엇을 만드는지에 따라 다르다. interface는 object를 정의할 때 사용하고 types alias는 새로운 함수를 생성할 때(call signatures를 이야기하는듯) 사용하는 것이 좋다.

1import React from "react";
2
3interface UserState {
4 name: string;
5 age: number;
6 address: string;
7}
8
9interface IUserListProps {
10 state: UserState[];
11}
12
13const UserList: React.FC<IUserListProps> = ({ state }) => {
14 return state.map(item => (
15 <li>
16 <span>{item.name}</span>
17 <span>{item.age}</span>
18 <span>{item.address}</span>
19 </li>
20 ));
21};
1interface Person {
2 name: string;
3 age: number;
4}
5
6type User = (person: Person) => Person;
7
8const user: User = person => person;
9
10user({ name: "Thor", age: 10000 });

Typescript 사용하기

타입스크립트를 사용하기 위해서 타입스크립트를 설치해야한다. 만약 CRA를 통해 React와 타입스크립트를 사용한다면 타입스크립트를 처음부터 세팅할 일을 거의 없다. 모든 환경 설정을 외울 필요는 없지만 간단하게 설정하는 방법을 정리해본다.

설치

폴더를 만들고 폴더 안에서 typescript를 설치한다.

1$ mkdir typescript-excercise
2$ cd typescript-excercise
3$ npm init -y
4$ npm install -D typescript

index.ts 파일 만들기

꼭 index.ts 파일을 만들 필요는 없다. 파일의 이름은 자유롭게 하면 된다.

1$ mkdir src/index.ts

index.ts파일에 간단하게 코드를 작성한다. 위에서 작성했던 코드를 하나 가져와서 붙여넣었다.

1interface User {
2 firstName: string;
3 lastName: string;
4 email: string;
5 getEamil(): void;
6 getFullName(): string;
7}
8
9interface Auth {
10 auth: string;
11}
12
13class Player implements User, Auth {
14 constructor(
15 public firstName: string,
16 public lastName: string,
17 public email: string,
18 public auth: string
19 ) {}
20 getEamil() {
21 console.log(this.email);
22 }
23 getFullName() {
24 return `${this.firstName} ${this.lastName}`;
25 }
26}
27
28const hyun = new Player("hyun", "san", "sanhyun@naver.com", "lev.1");

package.json

1{
2 // 생략
3 "scripts": {
4 "build": "tsc"
5 },
6 //생략
7 "devDependencies": {
8 "typescript": "^4.6.4"
9 }
10}

scripts에 build를 위와 같이 작성한다. npm build를 누르면 ts파일을 js로 컴파일 해준다. 지금은 동작하지 않는다.

tsconfig.json

tsconfig.json 파일을 만든다.

1$ touch tsconfig.json
1{
2 "include": ["src"],
3 "compilerOptions": {
4 "outDir": "build",
5 "target": "ES6",
6 "lib": ["ES6"],
7 "strict": true,
8 "allowJs": true,
9 "esModuleInterop": true,
10 "module": "CommonJS"
11 }
12}

더 많은 옵션이 있다. 필요할때 마다 살펴보기로 하자.

Declaration Files

많은 패키지가 자바스크립트로 작성되어있다. 그래서 자바스크립트로 작성된 함수가 어떤 모양인지 타입스크립트에 알려주어야할 수 있다. 그럴 때 *.d.ts파일을 작성하여 타입스크립트에게 설명을 해주어야한다.

tsconfig.ts파일에 "strict": true 를 설정하면 ‘Could not find a declaration file for module …’ 메시지를 볼 수 있다. *.d.ts 파일이 없기 때문인데 그럴 때 패키지 이름.d.ts로 파일을 만들고 자바스크립트 라이브러리의 call signatures를 작성하면 타입스크립트는 이 파일을 보고 자동 완성 기능등을 제공해준다. 작성은 아래처럼 하면 된다.

1declare module "mypackage" {
2 function init(value: string): void;
3}

아마 리액트 스타일 컴포넌트를 사용해봤다면 styled.d.ts 파일을 작성해본 경험이 있을 것이다.

JSDoc

만약에 이미 자바스크립트로 작성된 라이브러리가 있다면 그리고 코드 양이 방대하다면 타입스크립트에서 자바스크립트 코드를 사용할 수 있다. tsconfig.json에서 allowJs를 허용해주면 된다. 하지만 자바스크립트는 typescript의 보호를 받지 못한다. 그럴때 JSDoc 주석을 사용하여 자바스크립트 파일에 타입 정보를 제공할 수 있다. 주석이기 때문에 자바스크립트에서는 동작하지 않는다.

코드 출처 : 노마드 코더 타입스크립트로 블록체인 만들기

1// @ts-check
2/**
3 * Initialize the project
4 * @param {object} config
5 * @param {boolean} [config.debug]
6 * @param {string} config.url
7 * @returns {boolean}
8 */
9export function init(config) {
10 return true;
11}
12
13/**
14 * Exits the program
15 * @param {number} code
16 * @returns {number}
17 */
18export function exit(code) {
19 return code + 1;
20}

ts-node

ts-node는 매번 타입스크립트 파일을 빌드하지 않고 타입스크립트를 실행시켜준다. dev 환경에서 사용하면 편리하다. nodemon과 함께 사용하면 ts파일이 변경될 때 마다 다시 실행해준다.

1$ npm install ts-node nodemon -D
1// package.json
2{
3 "script": {
4 "dev": "nodemon --exec ts-node src/index"
5 }
6}

DefinitelyTyped

외부 패키지를 사용하는 경우가 많다. 하지만 패키지가 자바스크립트로만 작성되어있는 경우가 많은데 그럴 경우에는 에러 메시지에서 추천하는 방법으로 해결하는 것이 좋다. 거의 대부분은 npm i -D @types/**를 실행하라고 알려준다.

DefinitelyTyped는 types 정의를 모아놓은 깃 저장소다.

부록

참조 : 타입스크립트의 never 타입 완벽 가이드

never가 뭔지는 잘 모르지만 가끔 never[]에는 할당할 수 없다는 에러를 만난다. 그래서 우연하게 발견한 아티클을 읽고 never에 대한 대략적인 것을 정리하였다.

타입은 가능한 값의 집합이다. 타입스크립트에서 never타입은 값의 공집합이다. 집합에 어떠한 값도 없기 때문에, never 타입은 any 타입의 값을 포함해 어떤 값도 가질 수 없다.

1delcare const any : any
2const never:never = any //error

never 타입이 왜 필요할까?

숫자 체계에 아무것도 없는 양을 나타내는 0처럼 문자 체계에도 불가능을 나타내는 타입이 필요하다.

타입스크립트에서 나타내고 있는 불가능

유니언 교차 타입과 never의 동작

타입 필터링을 할 때 never를 사용하는 예제는 어쩌면 유용할지도 모른다는 생각이 든다. 나의 코드에 한번 정도 적용하기 위해서 노력해봐야겠다.

never 타입은 어떻게 읽을까?

명시적으로 never를 사용하지 않은 코드에서 never 타입과 관련된 오류 메시지를 받아본적이 있을 수 있다. 타입스크립트가 일반적으로 타입을 교차하기 때문이다. 문자열, 숫자, 불리언 교차 타입은 서로 호환되지 않기 때문에 never타입이 되고, 이것은 오류 메시지에 never가 표시되는 이유다.

이 문제를 사용하기 위해 타입 단언(또는 함수 오버로드)을 사용해야한다.