__proto__ 속성과 프로토타입 위임 과정

February 05, 2016

1. 글에 대해

  • 이 글은 (JS 위임 과정의 이해를 돕기 위한 글이 아닌)최근 시작한 ES567 스터디 그룹의 연구 과제를 위해 작성된 글이다.

    • 그 이해를 돕기위한 내용은 이 을 참고할 수 있다.

  • 즉 글 전반적으로 서술된 글이 아닌, 테스트 코드와 그에 대한 주석을 통해 설명하고있다.

2. 위임(과정) 이란?

  • 결국 객체들간의 원형 복제 과정을 말한다.(이 말을 잘 기억해두길 바란다)

    • (생성자)함수 (객체)를 통한 위임 과정

      // global execution context
      
      // A 함수 객체를 선언한다.
      function A(){
      
          // function execution context
      
      }
      
      // A 함수 객체의 prototype 프로퍼티로 포함된 원형 객체(이후 newAObject 객체로 위임될 객체이기도 하다)
      console.log(A.prototype);
      
      // new 연산자와 A (생성자)함수 (객체)를 통해 새로운 객체가 생성된다.
      var newAObject = new A;
      
      // newAObject 객체의 __proto__ (내부)속성은, A.prototype (원형)객체를 참조하고있다.
      console.log(newAObject.__proto__ === A.prototype); // true
      
      // newAObject 객체의 (생성자)함수인 A 함수 (객체)를 반환한다.
      console.log(newAObject.constructor); // A function object
      console.log(newAObject.constructor === A.prototype.constructor); // true
      
      // newAObject 객체의 prototype chain 에는 A.prototype 객체가 존재하므로 아래 코드를 통해 true 를 반환하게 된다.
      
      // instanceof 연산자는 왼쪽 파라메터로 전달된 객체(newAObject)의 prototype chain 내부에, 오른쪽 파라메터로 전달된 함수 객체(A)의 prototype 프로퍼티가 존재한다면 true 를 반환하게된다.
      console.log(newAObject instanceof A); // true
      
      console.dir(newAObject);

      class 00

      • 정리

        • 선언된 A 함수 객체는 prototype 이라는 (원형)객체를 가지고 있다.

        • new 연산자를 통해 객체 생성시, A.prototype 객체는 newAObject 객체의 __proto__ (내부)속성으로 위임된다.

        • 예상했던 결과와 같이, newAObject.constructor 속성은 (생성자)함수인 A 함수 객체를 참조하고 있다.

      instanceof 연산자

    • 객체(Object)를 통한 위임 과정

      // A 객체를 생성한다.
      var A = {
          id: '',
          getId: function(){
              return this.id;
          },
          setId: function(id){
              this.id = id;
          }
      };
      
      // A.id 값을 초기화한다.
      A.setId('yanione');
      
      // B 함수 객체를 선언한다.
      function B(name){
      
          name = name || '';
      
          this.name = name;
      }
      
      // B.prototype 객체 프로퍼티에 getName 메서드를 할당한다.
      B.prototype.getName = function(){
          return this.name;
      };
      
      // B.prototype 객체가 가진 맴버들을, 위임할 객체(_A Object)에 추가하기위해, (임시)변수에 저장해둔다.
      var __prototype__ = B.prototype;
      
      // 위임할 A 객체를 복제한다.
      var _A = {
          id: A.id,
          getId: A.getId
      };
      
      // 복제된 _A 객체를 B.prototype 객체 프로퍼티로 위임한다.
      B.prototype = _A;
      
      // __prototype__ 변수에 저장된 객체 맴버를, B.prototype 객체로 재 할당한다.
      B.prototype.getName = __prototype__.getName;
      
      //B.prototype.constructor = B;
      
      // newBObject 객체를 생성한다.
      var newBObject = new B('mohwa');
      
      // newBObject.__proto__ 속성은 B.prototype 객체를 참조하고 있다.
      console.log(newBObject.__proto__ === B.prototype); // true
      
      // B.prototype 객체 프로퍼티는 복제된 _A 객체를 참조하고 있다.
      console.log(newBObject.__proto__ === _A); // true
      
      console.log(newBObject.getId()); // yanione
      console.log(newBObject.getName()); // mohwa
      
      // instanceof 연산자 테스트
      try {
          // instanceof 연산자의 오른쪽 파라메터에는 반드시 함수 객체가 위치해야한다.
          console.log(newBObject instanceof _A); // error
      }
      catch(e){
          // 에러가 발생한다.
          console.log(e.message); // Expecting a function in instanceof check, but got #<Object>
      }
      
      // newBObject 객체의 prototype chain 에는 B.prototype 객체가 존재하므로 아래 코드를 통해 true 를 반환하게 된다.
      console.log(newBObject instanceof B); // true
      
      // _A 객체의 위임 과정으로인해, newBObject.constructor 속성은 Object 함수 객체를 반환하게된다.
      console.log(newBObject.constructor); // Object function object

      class 01

      • 정리

        • _A 객체위임 과정으로인해, newBObject.constructor 속성은 (생성자)함수인 B 함수 객체가 아닌, Object (생성자)함수 객체를 참조하게된다.

        • 위 상황을 우회(명확한 해결책은 아닌)하기 위해, 아래와 같은 코드를 추가할 수 있다.

          // B.prototype 객체에 constructor 속성을 추가시킨다.
          B.prototype.constructor = B;
    • new 연산자를 통한 위임 과정

      // A 함수 객체를 선언한다.
      function A(id){
      
          id = id || '';
      
          this.id = id;
      }
      
      A.prototype.getId = function(){
          return this.id;
      };
      
      
      // B 함수 객체를 선언한다.
      function B(name){
      
          name = name || '';
      
          this.name = name;
      }
      
      B.prototype.getName = function(){
          return this.name;
      };
      
      var __prototype__ = B.prototype;
      
      var newAObject = new A('yanione');
      
      // newAObject 객체를 B.prototype 객체 프로퍼티에 할당한다.
      B.prototype = newAObject;
      
      //B.prototype.constructor = B;
      
      B.prototype.getName = __prototype__.getName;
      
      var newBObject = new B('mohwa');
      
      // newBObject.__proto__.__proto__ 속성은 A.prototype 객체를 참조하고있다.
      console.log(newBObject.__proto__.__proto__ === A.prototype); // true
      
      // 즉 newBObject.__proto__.__proto__(A.prototype) 속성은 newAObject.__proto__(A.prototype) 속성과 동일하다.
      console.log(newBObject.__proto__.__proto__ === newAObject.__proto__); // true
      
      console.log(newBObject.getId()); // ''
      console.log(newBObject.getName()); // mohwa
      
      // newBObject 객체의 prototype chain 에는 B.prototype 객체가 존재하므로 true 를 반환하게된다.
      
      // newBObject.__proto__ === B.prototype
      console.log(newBObject instanceof B); // true
      
      
      // newBObject 객체의 prototype chain 에는 A.prototype 객체도 존재하므로 true 를 반환하게된다.
      
      // newBObject.__proto__.__proto__ === A.prototype
      console.log(newBObject instanceof A); // true
      
      // new A 객체의 위임 과정으로인해 newBObject.constructor 속성은 A 함수 객체를 참조하게된다.
      console.log(newBObject.constructor); // A function object
      
      console.log(newBObject);

      class 02

      • 정리

        • newBObject 객체 prototype chain 에는 A.prototype 객체B.prototype 객체 프로퍼티가 모두 존재하며, instanceof 연산자를 통해 모두 true 를 반환하게된다.

        • new A 객체의 위임 과정으로인해, newBObject.constructor 속성은 B (생성자)함수 객체가 아닌, A 함수 객체를 참조하게된다.

    • Object.create 메서드를 통한 위임 과정

      // A 함수 객체를 선언한다.
      function A(id){
      
          id = id || '';
      
          this.id = id;
      }
      
      A.prototype.getId = function(){
          return this.id;
      };
      
      
      // B 함수 객체를 선언한다.
      function B(name){
      
          name = name || '';
      
          this.name = name;
      }
      
      B.prototype.getName = function(){
          return this.name;
      };
      
      var __prototype__ = B.prototype;
      
      // Object.create 메서드를 통해 A.prototype 객체가 위임된, 새로운 객체가 생성된다.
      /*
      
          {
             __proto__: A.prototype
          }
      */
      var newObject = Object.create(A.prototype);
      
      // B.prototype 객체에 Object.create 메서드를 통해 생성된 객체를 할당한다.
      B.prototype = newObject;
      
      //B.prototype.constructor = B;
      
      B.prototype.getName = __prototype__.getName;
      
      var newBObject = new B('mohwa');
      
      console.log(newBObject.getId()); // undefined
      console.log(newBObject.getName()); // mohwa
      
      // Object.create 메서드를 통한 위임 과정으로인해, newBObject.constructor 속성은 A 함수 객체를 참조하게된다.
      console.log(newBObject.constructor); // A function object
      
      console.log(newBObject);

      class 03

      • 정리

        • Object.create 메서드를 통해 new 연산자를 통해서만 가능했던 위임 과정을 구현했다.

        • Object.create 메서드를 통한 위임 과정으로인해, newBObject.constructor 속성은 B (생성자)함수 객체가 아닌, A 함수 객체를 참조하게된다.

        • Object.create 메서드를 통한 위임 과정으로인해, 인스턴스 맴버인 id 속성을 초기화하지 못했다.(반드시 new 연산자를 통해서만 인스턴스 맴버를 초기화할 수 있다)

    • (인스턴스 맴버를 초기화하는)Object.create 메서드를 통한 위임 과정

      // A 함수 객체를 선언한다.
      function A(id){
      
          id = id || '';
      
          this.id = id;
      }
      
      A.prototype.getId = function(){
          return this.id;
      };
      
      
      // B 함수 객체를 선언한다.
      function B(id, name){
      
          id = id || '';
          name = name || '';
      
          this.name = name;
      
          // call 메서드를 통해 A (생성자)함수 객체가 가진 인스턴스 맴버를 초기화한다.
          A.call(this, id);
      }
      
      B.prototype.getName = function(){
          return this.name;
      };
      
      var __prototype__ = B.prototype;
      
      var newObject = Object.create(A.prototype);
      
      // B.prototype 객체에 Object.create 메서드를 통해 생성된 객체를 할당한다.
      B.prototype = newObject;
      
      //B.prototype.constructor = B;
      
      B.prototype.getName = __prototype__.getName;
      
      var newBObject = new B('yanione', 'mohwa');
      
      console.log(newBObject.getId()); // yanione
      console.log(newBObject.getName()); // mohwa
      
      console.log(newBObject);

      class 04

      • 정리

        • call 메서드를 통해 B (생성자)함수 객체 호출 시 생성된 인스턴스 맴버로, A (생성자) 함수 객체의 인스턴스 맴버id 속성이 초기화되었다.

관련 URL


Profile picture

Written by mohwa