본문 바로가기
Computer Science/Javascript

JavaScript - 데이터 타입과 연산자(객체타입 / 참조 타입의 특성)

by 마늘아빠 2021. 4. 13.
728x90
반응형
자바스크립트 참조 타입(객체 타입)

자바스크립트에서 숫자, 문자열, 불린 값, null, undefined 같은 기본 타입을 제외한 모든 값은 객체다.

즉, 배열, 함수, 정규표현식 모두 객체로 표현된다.

자바스크립트에서 객체는 단순히 '이름(key): 값(value)' 형태의 프로퍼티들을 저장하는 컨테이너로서, 해시 자료구조와 유사하다. 자바스크립트에서 기본 타입은 하나의 값만을 가지는 데 비해, 참조 타입인 객체는 여러 개의 프로퍼티들을 포함할 수 있으며, 이러한 객체의 프로퍼티는 기본 타입의 값을 포함하거나, 다른 객체를 가리킬 수도 있다. 이러한 프로퍼티의 성질에 따라 객체의 프로퍼티는 함수로 포함할 수 있으며, 자바스크립트에서는 이러한 프로퍼티를 메서드라고 부른다.

 

객체 생성

자바스크립트의 객체 개념은 생성 방법이나 상속 방식 등에서 C++ 혹은 자바와 같은 기존 객체 지향 언어의 객체 개념과 약간 다르다. 자바에서는 클래스를 정의하고, 클래스의 인스턴스를 생성하는 과정에서 객체가 만들어진다. 이에 비해 자바스크립트에서는 클래스라는 개념이 없고, 객체 리터럴이나 생성자 함수 등 별도의 생성 방식이 존재한다.

자바스크립트에서 객체를 생성하는 방법은 크게 세 가지가 있다. 

1. Object() 객체 생성자 함수를 이용하는 방법

2. 객체 리터럴을 이용하는 방법

3. 생성자 함수를 이용하는 방법

 

1. Object() 객체 생성자 함수를 이용하는 방법

자바스크립트에서는 객체를 생성할 때, 내장 Object() 생성자 함수를 제공한다. 


// Object()를 이용해서 foo 빈 객체 생성
var foo = new Object();

// foo 객체 프로퍼티 생성
foo.name = 'foo';
foo.age = 30;
foo.gender = 'male';

console.log(typeof foo);
console.log(foo);

출력 결과

Object() 생성자 함수를 이용해서 foo라는 빈 객체를 생성한 후, 몇 가지 프로퍼티들을 추가했다.


 

2. 객체 리터럴을 이용하는 방법

리터럴이란 표기법이라고 생각하면 되는데, 객체 리터럴이란 객체를 생성하는 표기법을 의미한다.

객체 리터럴은 중괄호를 이용해서 객체를 생성한다. {} 안에 아무것도 적지 않으면 빈 객체가 생성된다.

"프로퍼티 이름" : "프로퍼티 값" 형태로 표기하면 해당 프로퍼티가 추가된 객체를 생성할 수 있다.

어떤 표현식도 프로퍼티 값이 될 수 있으며, 만약 함수가 된다면 이러한 프로퍼티를 메서드라고 부른다.


// 객체 리터럴 방식으로 foo 객체 생성
var foo = {
     name: 'foo',
     age: 30,
     gender: 'male',
};

console.log(typeof foo);
console.log(foo);

출력결과


 

3. 생성자 함수를 이용하는 방법

// 객체 리터럴 방식으로 foo 객체 생성
var foo = {
     name: 'foo',
     age: 30,
     gender: 'male',
};

console.log(foo);

// 생성자 함수
function Person(name, age, gender, position) {
     this.name = name;
     this.age = age;
     this.gender = gender;
}

// Person 생성자 함수를 이용해 bar 객체, baz 객체 생성
var bar = new Person('bar', 33, 'woman');
console.log(bar);

var baz = new Person('baz', 25, 'woman');
console.log(baz);

출럭 결과

리터럴 타입과 달리 Person 객체로 명시되어있는 것을 확인할 수 있다.


 

객체 프로퍼티 읽기/쓰기/갱신

객체는 새로운 값을 가진 프로퍼티를 생성하고, 생성된 프로퍼티에 접근해서 해당 값을 읽거나 또는 원하는 값으로 갱신할 수 있다.

객체의 프로퍼티에 접근하려면 다음과 같이 두 가지 방법을 사용한다.

  • 대괄호( [ ] ) 표기법
  • 마침표(. ) 표기법

// 객체 리터럴 방식을 통한 foo 객체 생성
var foo = {
     name: 'foo',
     major: 'computer science',
};

// 객체 프로퍼티 읽기
console.log(foo.name);
console.log(foo['name']);
console.log(foo.nickname); // 없는 프로퍼티. undefined

// 객체 프로퍼티 갱신
foo.major = 'electronics engineering';
console.log(foo.major);
console.log(foo['major']);

// 객체 프로퍼티 동적 생성
foo.age = 30;
console.log(foo.age);

// 대괄호 표기법만을 사용해야하는 경우
foo['full-name'] = 'foo bar';
console.log(foo['full-name']);
console.log(foo.full - name); // 빼기로 인식, 그래서 숫자가 아님 NaN
console.log(foo.full);
console.log(name); // 없는 변수. undefined

출력 결과

위의 결과에서 foo.age = 30에 주목해보자. 이는 프로퍼티 동적 생성으로 해당 객체에 프로퍼티를 추가한 것이다.

즉, 자바스크립트에서 객체의 프로퍼티에 값을 할당할 때, 프로퍼티가 이미 있는 경우는 값 갱신이 되고, 없는 경우에는 동적으로 생성된 후 값이 할당된다.

프로퍼티 이름에 연산자가 들어가 있는 경우에는 대괄호 표기법만 사용해야 한다. 

NaN ( Not a Number ) 값

수치 연산을 해서 정상적인 값을 얻지 못할 때 출력되는 값이다. 1 - 'foo'가 NaN인 것.


 

for in 문과 객체 프로퍼티 출력

for in 문을 사용하면, 객체에 포함된 모든 프로퍼티에 대해 루프를 수행할 수 있다. 


// 객체 리터럴을 통한 foo 객체 생성
var foo = {
     name: 'foo',
     age: 30,
     major: 'computer science',
};

// for in 문을 이용한 객체 프로퍼티 출력
var prop;
let idx = 0;
for (prop in foo) {
     idx++;
     console.log(idx, prop, foo[prop]);
}

출력 결과

for in 문이 수행되면서 prop 변수에 foo 객체의 프로퍼티가 하나씩 할당된다. 따라서 prop에 할당된 프로퍼티 이름을 이용해서 foo [prop]와 같이 대괄호 표기법을 이용해서 프로퍼티 값을 출력했다.


 

객체 프로퍼티 삭제

객체 프로퍼티를 delete 연산자를 이용해 즉시 삭제가 가능하다. 객체 자체를 삭제하는 것이 아닌, 프로퍼티를 삭제하는 것이다.


// 객체 리터럴을 통한 foo 객체 생성
var foo = {
     name: 'foo',
     nickname: 'babo',
};

console.log(foo.nickname);
delete foo.nickname;
console.log(foo.nickname);

delete foo;
console.log(foo);

츨력 결과



참조 타입의 특성

객체는 참조 타입라고 부르는데 그 이유는 모든 연산이 참조값으로 처리되기 때문이다.


var objA = {
     val: 40,
};

var objB = objA; // 같은 객체를 가리킴

console.log(objA.val); // 40
console.log(objB.val); // 40

objB.val = 50;

console.log(objA.val); // 50 변경
console.log(objB.val); // 50 변경

항상 참조 값으로 연산이 일어남을 기억하자!


객체 비교

동등 연산자(==)를 사용하여 두 객체를 비교할 때도 객체의 프로퍼티 값이 아닌 참조 값을 비교한다!


var a = 100;
var b = 100;

var objA = { val: 100 };
var objB = { val: 100 };
var objC = objB;

console.log(a == b); // 기본타입. 같은 값 true
console.log(objA == objB); // 참조타입. 다른 객체 false
console.log(objB == objC); // 참조타입. 같은 객체 true

값 비교와 참조 비교를 주의하자


 

참조에 의한 함수 호출 방식

기본타입 : 값에 의한 호출 ( Call by Value )

기본 타입의 값이 인자가 되면 매개변수로 복사된 값이 넘어간다. 그렇기 때문에 함수 내부에서 아무리 값을 바꾸어도 실제 호출된 변수의 값이 변하지는 않는다.

참조타입 : 참조에 의한 호출 ( Call by Reference )

참조 타입이 객체로 전달이 되면, 객체의 프로퍼티가 복사가 되는 것이 아니라 실제 객체의 참조 값이 전달된다. 때문에 함수 내부에서 값 변경이 일어난다면 실제 객체의 값이 변경이 된다.


var a = 100;
var objA = { val: 100 };
function changeArg(num, obj) {
     num = 200; // 함수 내부, Call by Value. 실제 값 변경 X
     obj.val = 200; // Call by Reference. 호출 된 객체의 값 변경

     console.log(num, obj);
}

changeArg(a, objA);
console.log(a, objA);

출력 결과

Call by Value와 Call by Reference의 동작 차이를 확실히 알아야한다.


 

반응형