[Typescript] 객체와 타입

박솔찬·2021년 8월 26일
0

TypeScript

목록 보기
1/2
post-thumbnail

객체와 타입

자바스크립트가 아닌 타입스크립트의 객체와 타입의 개념과 관련 구문을 살펴보자.

변수 선언방법

타입스크립트의 기본 제공 타입

타입스크립트는 자바스크립트와 호환성을 위해서 자바스크립트의 타입과 타입스크립트의 타입을 모두 지원한다.

자바스크립트타입스크립트
Numbernumber
Booleanboolean
Stringstring
Objectobject

변수 선언하기

es5 자바스크립트는 var키워드를 통해 변수를 선언하였다. 하지만 var는 다른 프로그래밍 언어와는 다르게 동작한다. 자세한 문제는 아래 링크를 참고하자.
var 키워드를 사용하면 안되는 이유!

그래서 es6(ESNext)부터 let과 const 키워드가 추가되었다.
let 키워드는 수시로 코드에서 값이 변경될 수 있고, 다른 프로그래밍 언어의 일반적인 변수 선언 방식과 같다.
const 키워드는 한 번 선언한 변수값은 변경될 수 없고, 선언시 초기화가 같이 이루어져야 한다.

let num = 123; // num: 123
num = 321; // num: 321

const name = 'solchan' // name: solchan
name = 'boy' // name: solchan, 불가능

타입스크립트에서는 변수를 선언할 때, 타입을 지정할 수 있다.

let 변수명: 타입 = 초기값
const 변수명: 타입 = 초기값
let num: number = 123; // num: 123
num = 321; // num: 321
num = 'one'; // num: 123, string은 불가능

const name: string = 'solchan' // name: solchan
const age: number = '24'; // X, number타입에 string 불가능

하지만 타입스크립트는 자바스크립트와 호환성을 위해 아래와 같이 선언하여도 타입 추론이 가능합니다.
이를 타입 추론이라고 합니다.

let num = 123; // 123을 통해 number로 타입 추론
const name = 'solchan' // 'solchan'을 통해 string으로 타입 추론

혹은, any타입을 지정할 수 있습니다.
any타입은 타입을 지정하지 않은 경우와 같이 동작한다.

let val: any = 123; // 지금은 숫자
val = {}; // 지금은 객체
val = 'solchan'; // 지금은 문자열

마지막으로 undefined에 대해 알아보면, 타입스크립트에서는 타입과 값으로 동작한다.
아래 예시를 보고 이해해보자

let val: undefined = undefined; // 타입: undefined, 값: undefined

val = 1; // 불가능, val 타입이 undefined이기 때문에 number는 불가능

인터페이스

타입스크립트에서 object타입으로 선언된 변수는 number, boolean, string 타입의 값을 가질 수는 없지만, 속성이 다른 object에 대해서는 마치 any타입처럼 담겨집니다.

let obj: object = { name: 'solchan'};
obj = { name: 'sochan', age: 24};

1번 코드는 name만을 갖는 object이고, 2번 코드는 name, age를 갖는 object입니다. 서로 속성이 다르지만, object로 타입이 같아 위 코드가 가능합니다.

이런 부분을 해결하기 위해 인터페이스 구문이 고안되었고, 아래와 같은 방법으로 사용이 됩니다.

interface 이름 {
    속성: 타입
}

예시

// User
interface User {
    name: string;
    age: number;
    phone?: string; // 변수명 + ? 는 선택 항목을 뜻한다.
}

위 인터페이스를 통해 예시를 보면,

let user1: User = { name: 'solchan', age: 24 }
// user1 -> 필수 항목인 name과 age 모두 초기화, 성공!
let user2: User = { name: 'solchan', phone: '123-123' }
// user2 -> 팔수 항목인 age가 없엄, 오류!
let user3: User = { name: 'solchan', age: 24, location: 'YongIn' }
// user3 -> 필수 항목이 모두 있지만 location은 인터페이스에 없음, 오류!
let user4: User = { name: 'sochan', age: 24, phone: '123-123' }
// user4 -> 필수가 모두 존재하고 phone은 선택으로 존재한다, 성공!

마지막으로 익명으로 인터페이스 사용하기.

// 위 User인터페이스를 사용하지 않는다.
let account: { email: string, password: string } = { email: 'solchan@interface.hello', password: 'typescript' }

익명 인터페이스는 주로 함수에서 사용된다.

function printMe( me: { name: string, age: number }) {
    console.log( `name: ${me.name}, age: ${me.age}` )
}

printMe({ name: 'solchan', age: 24});

클래스

타입스크립트는 C++, Java와 같이 객체지향언어이다.
따라서 class, 접근 제한 키워드 private, protected, public 그리고 implements, extend와 같은 키워드를 제공한다.

선언 방법

클래스 선언 방법은 다음과 같다.

class 이름 {
    접근제한자 속성: 타입
}
// 접근제한자를 지정하지 않으면, 기본으로 public이 지정된다.

constructor라는 생성자가 있는데, 객체 생성을 다음과 같이 할 수 있다.

/* name과 age를 갖는 user 객체를 만들어 보자  */
/* 아래 여러 방법으로 다음과 같은 객체를 생성할 수 있다. */
const user: User = new User('solchan', 24);
console.log(user);
// 출력 ->  User { name: 'solchan, age: 24}

/* 1번 */
class User {
    constructor(name: string, age: number) {
        this.name = name;
        this.age = age;
    }
    name: string;
    age: number;
}
/* 2번 */
/* 생성자 파라미터 부분에 아래처럼 하여 속성 정의를 생략할 수 있다. */
class User {
    constructor(public name: string, public age: number) {}
}

인터페이스 구현

정의한 인터페이스를 구현하는 클래스를 만들어 보자.

// 이름과 나이를 정의하는 인터페이스를 상속하는 클래스를 만들어 보자.

interface User {
    name: string;
    age?: number;
}

class UserClass {
    constructor(public name: string, public age: number){]}
}
const user: UserClass = new UserClass('solchan', 24);
console.log(user);
// 출력 -> UserClass { name: 'solchan', age: 24 }

여기서 interface의 age는 필수가 아닌 선택으로 정의되었지만, 클래스가 인터페이스를 구현하는 경우에는 반드시 속성으로 포함되어야 한다.

마지막으로, 추상 클래스와 상속에 대해 알아보자.
타입스크립트는 다른 객체지향 언어처럼 abstract 키워드를 통해 아래와 같이 추상 클래스를 만들어 사용할 수 있다.

abstract class 이름 {
    abstract 속성: 타입;
    abstract 메서드(): {};
}

예시

abstract class AbstractUser {
    abstract name: string;
    constructor(public age: number) {};
}
// abstract 키워드를 붙인 클래스 및 속성은 new를 통한 생성자를 통해 생성이 불가능하다.

위 추상 클래스를 상속받는 User 클래스를 만들어 보자.
super()는 부모 클래스의 생성자를 호출하는 것이다.

abstract class AbstractUser {
    abstract name: string;
    constructor(public age: number) {};
}
class user extends AbstractUser{
    constructor(public name: string, age: number){
        super(age);
    };
/* 부모 클래스의 name이 abstract이기 때문에, 구현 구문을 작성한다.
생성자 구문에 작성하였다. */
}

console.log(new user('solchan', 24));
// 출력 -> user { age: 24, name: 'solchan' }

객체 구조 분해 할당과 스프레드 연산자

ES6(ESNext)부터 추가된 구조 분해 할당과 스프레드 연산자를 알아보자.

구조 분해 할당 (비구조화 할당) / Destructuring

구조 분해 할당은 객체와 배열 모두 사용 가능하다.
여기서는 객체에 대한 구조 분해 할당을 알아보자

class User {
    constructor(
        public name: string, 
        public age: number, 
        public location: string){};
}
const user1 = new User('solchan', 24, 'Y');

const { age, name: nickname, location: loc } = user1;
console.log(`이름: ${nickname}, 나이: ${age}, 지역: ${loc}`)
// 출력 -> 이름: solchan 나이: 24 지역: Y

user1 객체에서 name을 nickname이라는 이름으로, location을 loc로, age는 그대로 꺼내어 마치 단독 변수처럼 사용할 수 있다.

스프레드 연산자

스프레드 연산자는 ...을 통해 필요없는 속성을 빼거나 다른 속성을 추가할 수 있습니다.
빼는 것을 Rest, 추가하는 것을 spread라고 한다.

Rest Operator

class User {
    constructor(
        public name: string,
        public age: number,
        address: string,
        country: string){
            this.address = address;
            this.country = country;
    };
    address: string;
    country: string;
}

const user = new User('solchan', 24, 'YongIn', 'Korea');

const { country, address, ...nameAgeUser } = user;
console.log(user);
console.log(nameAgeUser);
/*
출력(user) -> User { name: 'solchan', age: 24, address: 'YongIn', country: 'Korea' }
출력(nameAgeUser) -> { name: 'solchan', age: 24 }
*/

16번 코드를 보면 country, address를 user에서 빼고 이를 nameAgeUser로 만드는 동작을 한다.

출력 결과를 보면 nameAgeUser는 country, address가 빠진 상태로 출력됨을 알 수 있다.

user는 User이라고 클래스 명을 출력해준다.
하지만, nameAgeUser은 country와 address 속성이 빠졌기 때문에 User클래스가 되지 못한다.
따라서 속성만 출력이 되는 것을 알 수 있다.

Spread Operator

const part1: object = {
    name: 'solchan'
}

const part2: object = {
    country: 'Korea',
    address: 'YongIn',
}

const user = { ...part1, ...part2 };
console.log(part1);
console.log(part2);
console.log(user);
// 출력(part1) -> { name: 'solchan' }
// 출력(part2) -> { country: 'Korea', address: 'YongIn' }
// 출력(user) -> { name: 'solchan', country: 'Korea', address: 'YongIn' }

10번 코드를 보면, part1, part2를 합하여 user를 만드는 것을 알 수 있다.


본 게시글은 Typescript를 공부하며 정리된 내용이며,
'Do it! 타입스크립트 프로그래밍' 책을 참고하여 공부하고 있습니다.
잘못된 정보가 존재한다면 바로바로 수정할 것이며,
언제든 피드백은 환영입니다!

profile
Why? How? What?

0개의 댓글