Programming Language

JS33 - 16.new, 생성자, instanceof, 인스턴스

양찬우 2021. 6. 3. 19:18
728x90

new

  • 생성자 함수 (Constructor function)을 호출하는 방법이다.
  • 이름은 대문자로 시작한다.
  • new 신규작성 시,
    • 새로운 빈 오브젝트를 생성한다.
    • 새로 생성된 오브젝트에 this를 바인딩한다.
    • 새로 생성된 오브젝트에 생성자 함수의 prototype object를 가리키는 proto라는 속성을 추가한다.
    • 생성된 오브젝트가 함수로부터 반환될 수 있도록 함수의 마지막에 return this를 추가한다. (function에서 object를 반환)
function Student(name, age) {
  this.name = name;
  this.age = age;
}

var second = new Student('Jeff', 50);

second.__proto__ === Student.prototype;
// true

  • Student constructor 함수는 .prototype 속성을 갖고있다. 이 prototype은 다시 constructor를 가리키는 .constructor라는 오브젝트를 지니고 있다. (loop다!)
  • 이러한 속성과 오브젝트의 loop는 상속의 개념을 갖고 있기 때문에 중요하다. prototype 오브젝트는 constructor함수를 통해 해당 함수로 생성된 모든 오브젝트 간 공유된다. 즉, 모든 오브젝트가 사용해야할(필요한) 함수와 속성을 prototype에 추가할 수 있다는 뜻이다.
    • toString()같은 메소드를 직접 추가하지 않아도 사용할 수 있는 이유! (Object prototype에 자동생성)

생성자의 return

  • 일반적으로 생성자에는 return 문이 없습니다. 생성자의 목적은 this에 필요한 내용을 모두 적는 것이고, 그 자체로 목적입니다.
  • 하지만 return문이 있으면 아래와 같이 작동합니다.
    • object와 함께 return이 호출되면, object가 this대신 반환됩니다.
    • primitive와 함께 return이 호출되면 무시됩니다.
  • return + object -> object반환
  • 다른 경우 모두 this반환
// 1
function BigUser() {
  this.name = "John";
  return { name: "Godzilla" };  // <-- returns an object
}
alert( new BigUser().name );  // Godzilla, got that object ^^'

// 2
function SmallUser() {
  this.name = "John";
  return; // finishes the execution, returns this
  // ...
}
alert( new SmallUser().name );  // John

생성자의 메소드

  • 오브젝트를 생성하기 위해 생성자를 사용하면 매우 큰 유연성이 보장된다. 생성자 함수는 오브젝트를 정의하는 인자를 가질 수 있으며, 인자에 무엇이 들어갈 지도 정할 수 있다.
  • 물론 this에 속성뿐만 아니라 메소드도 넣을 수 있다.
  • 아래 예시에서 new User(name)은 주어진 name과 sayHi 함수를 지닌 채 생성된다.
function User(name) {
  this.name = name;

  this.sayHi = function() {
    alert( "My name is: " + this.name );
  };
}

let john = new User("John");

john.sayHi(); // My name is: John

/*
john = {
   name: "John",
   sayHi: function() { ... }
}
*/

요약

  • 생성자 함수의 첫 문자는 대문자로 생성한다.
  • 생성자 함수는 오직 new로만 호출한다. 이 호출은 비어있는 this를 생성하며 마지막에 생성된 오브젝트를 반환한다.
  • JS는 built-in 오브젝트에 생성자 함수를 제공한다. ex) Date, Set

DRY한 JS 객체 작성 방법

// 기존 방법

// 1. 개체문자표기법
let user1 = {
  name: "Taylor",
  points: 5,
  increment: function() {
    user1.points++;
  }
};

// 2. Object.create()
let user2 = Object.create(null);

user2.name = "Cam";
user2.points = 8;
user2.increment = function() {
  user2.points++;
}
  • 기존의 두 가지 방법은 반복적이며 항목을 수동으로 생성해야한다. 대안으로 아래와 같은 방법들이 있다.

1. function으로 오브젝트 생성

function createUser(name, points) {
  let newUser = {};
  newUser.name = name;
  newUser.points = points;
    // function으로 생성
  newUser.increment = function() { 
    newUser.points++;
  };
  return newUser;
}

let user1 = createUser("Bob", 5);
user1.increment(); 
// 단, 이 increment는 복사본일 뿐이다. 
// 각 개체에 대해 기능의 변경 가능성을 수동으로 수행해야 하므로 이 방법은 코드를 쓰는 데 적합하지 않다.

2. JS의 prototypal nature를 활용

function createUser(name, points) {
  let newUser = Object.create(userFunction);
  newUser.name = name;
  newUser.points = points;
  return newUser;
}

let userFunction = {
  increment: function() {this.points++};
  login: function() {console.log("Please login.")};
}

let user1 = createUser("Bob", 5);
user1.increment();
  • 모든 JavaScript 기능에는 기본적으로 비어 있는 프로토타입 속성이 있습니다. 이 프로토타입 속성에 기능을 추가할 수 있으며, 이 양식에서는 이를 메소드라고 합니다. 상속된 기능이 실행되면 이 값이 상속되는 개체를 가리킵니다.
  • user1 객체가 생성되었을 때 userFunction과의 프로토타입 체인 결합이 형성되었습니다.
  • user1.increment()가 call stack에 있는 경우, 글로벌 메모리에서 user1을 찾습니다. 다음으로, incrementfunction을 찾지만 찾지 못합니다. prototype chain의 다음 object를 보고 거기서 increment function을 찾을 것입니다.

3. new와 this 활용

function User(name, points) {
 this.name = name; 
 this.points = points;
}
User.prototype.increment = function(){
 this.points++;
}
User.prototype.login = function() {
 console.log(“Please login.”)
}

let user1 = new User(“Dylan”, 6);
user1.increment();

prototype과 proto의 차이

  • 프로토타입(Prototype)은 생성된 개체에서 proto 속성이 되는 것을 결정하는 생성자 기능의 속성입니다.
  • __proto__는 생성된 참조이며, 이 참조는 프로토타입 체인 결합이라고 알려져 있습니다.

4. ES6의 syntactic sugar

class User {
  constructor(name, points) {
    this.name = name;
    this.points = points;
  }
  increment () {
    this.points++;
  }
  login () {
    console.log("Please login.")
  }
}

let user1 = new User("John", 12);
user1.increment();
  • 3번과 동일하지만, ES6에서 도입된 class구문을 사용하면 더 깔끔하게 작성할 수 있다.

typeof vs instanceof

  • typeof는 값이 원시 타입의 요소인지 확인한다.
     if (typeof value === 'string')​
  • instanceof는 값이 클래스 또는 생성자 함수의 인스턴스인지 확인한다.
    if (value instanceof Map)​
  • 하지만 이것은 이상적이지 못하다. 왜냐하면 원시 값과 객체의 차이점이 종종 모호해지기 때문이다. 추가로 몇가지 이상한 점이 상황을 더욱 복잡하게 만든다.
    • typeof null은 'object'이다. 'null'이 아니다. 이것은 버그로 간주된다. (하지만 코드일관성을 위해 안 고침)
    • typoeof는 객체와 함수를 구분할 수 있다.
      > typeof {}
      'object'
      > typeof function () {}
      'function'
  • typeof를 통해 개체성을 확인할 수 있는 간단한 방법이 없다는 것을 의미한다.

원시타입에 instanceof 적용

PrimitiveNumber 클래스가 주어지면, 아래의 코드는 x instanceof PrimitiveNumber 표현식이 true를 반환하는 값 x를 구성한다. PrimitiveNumber에 대한 정적 메서드를 구현하면 키가 공용 심볼 인 Symbol.hasInstance가 된다.

class PrimitiveNumber {
    static [Symbol.hasInstance](x) {
        return typeof x === 'number';
    }
}
console.log(123 instanceof PrimitiveNumber); // true

instance 타입 지정

  • 특정 오브젝트가 다른 무언가의 instance 임을 확인하려면 instanceof를 사용하면 된다.
myBook instanceof Book    // true
myBook instanceof String  // false 
  • instanceof의 우측이 함수가 아니면 에러가 발생한다.
myBook instanceof {};
// TypeError: invalid 'instanceof' operand ({})
  • instance 의 타입을 확인하는 다른 방법은 constructor 속성을 사용하는 것이다.
    myBook.constructor === Book;   // true​
    • myBook의 생성자 속성은 Book을 가리키고 있기 때문에 === 연산이 true다. JS의 모든 오브젝트는 각자의 prototype에게서 constructor 속성을 상속받는다.
    • 단, constructor 속성을 통해 instance확인하는 것은 불필요하게 코드가 길어지기 때문에 안 좋은 방법이다.

Object.defineProperty() 메소드

  • 생성자 내부에서 Object.defineProperty() 를 사용하여 필요한 모든 속성을 설정할 수 있다.
    function Book(name) { 
      Object.defineProperty(this, "name", { 
          get: function() { 
            return "Book: " + name;       
          },        
          set: function(newName) {            
            name = newName;        
          },               
          configurable: false     
       }); 
    }
    
    var myBook = new Book("Single Page Web Applications");
    console.log(myBook.name);    // Book: Single Page Web Applications
    
    // we cannot delete the name property because "configurable" is set to false
    delete myBook.name;    
    console.log(myBook.name);    // Book: Single Page Web Applications
    
    // but we can change the value of the name property
    myBook.name = "Testable JavaScript";
    console.log(myBook.name);    // Book: Testable JavaScript​
  • 이 코드는 속성 접근자를 정의하기 위해 Object.defineProperty()를 사용했습니다. 속성접근자는 다른 어떠한 속성이나 메서드를 포함하지 않지만, 속성을 읽을때 호출되는 getter와 프로퍼티가 작성될 때 호출되는 setter를 정의합니다.
  • getter는 값을 리턴해야하며, setter는 속성에 할당된 값을 인자로 받는다. 위의 생성자는 name 속성이 변경/저장될 수 있는 instance를 반환하지만, 삭제는 할 수 없다. 그리고 우리가 이름의값을 얻으려할때 getter는 ‘Book: ‘이라는 문자열을 앞에 붙이고 리턴을한다.

객체리터럴 표기법은 생성자를 선호한다.

  • JS는 아홉 개의 만들어진 생성자가 있습니다: Object(), Array(), String(), Number(), Boolean(), Date(), Function(), Error(), RegExp()
  • 값을 만들 때, 객체리터럴이든 생성자든 아무거나 사용해도 된다.
  • 단, 객체리터럴이 더 읽기 쉽고 작동도 빠를 뿐만 아니라 파싱할 때 최적화가 가능하기 때문에 간단한 오브젝트는 그냥 객체 리터럴을 사용하자.
  • 리터럴에서도 메소드를 호출할 수 있다.
    • 메소드가 오브젝트에서 호출될 때, JS는 리터럴을 임시 오브젝트로 변환하고 작동가능하도록 만든다. 임시 오브젝트가 불필요해지면 그 때 삭제된다.

new 키워드 꼭 쓰자

  • 모든 생성자 만들 때 new 꼭 쓰자.
  • 실수로 new안 쓰면 새롭게 생성하려고 했던 오브젝트 대신 global object를 (ex window) 수정하게 된다.
function Book(name, year) {
  console.log(this);
  this.name = name;
  this.year = year;
}

// new 안쓰면 window가 수정된다...!
var myBook = Book("js book", 2014);  
console.log(myBook instanceof Book);  //false
console.log(window.name, window.year); // js book 2014

var myBook = new Book("js book", 2014);  
console.log(myBook instanceof Book); // true
console.log(myBook.name, myBook.year); // js book 2014

Scope-safe 생성자

  • 생성자는 단순히 함수다. 즉 new없이 호출 가능하다.
  • 단, 제대로 사용하지 않으면 (바로 위 단락처럼) 의도치 않은 오류가 발생한다.
  • scope-safe 생성자를 사용하면 new가 있든 없든 같은 결과를 내도록 도와준다.
  • Object,Regex, Array같은 내장함수는 모두 scope-safe다.
    • new사용 안하면 해당 생성자를 다시 new로 호출해서 올바른 인스턴스를 반환한다.
function Fn(argument) { 

  // if "this" is not an instance of the constructor
  // it means it was called without new  
  if (!(this instanceof Fn)) { 
    // call the constructor again with new
    return new Fn(argument);
  } 
}
// scope-safe 버전
function Book(name, year) { 
  if (!(this instanceof Book)) { 
    return new Book(name, year);
  }
  this.name = name;
  this.year = year;
}

var person1 = new Book("js book", 2014);
var person2 = Book("js book", 2014);

console.log(person1 instanceof Book);    // true
console.log(person2 instanceof Book);    // true
728x90