본문 바로가기
개발/웹

javascript 내장함수와 closure [펌]

by 로그인시러 2017. 7. 4.

난 자바스크립트에서 객체지향이란건 생각해본적도 없다. 하지만 클로저(Closure) 라는 패턴(?)을 이용해 자바(java)에서 쓰이는 클래스개념을 만들 수가 있다. 그렇다고 자바처럼 단순히 멤버변수와 메소드를 정의만 하는 쉬운방법으로는 되지 않는다. 우선 Closure 가 무엇인지 알기전에 내장함수(inner function)에 대해 알아보자


function outer(num1) {

	var num2 = 5;
    
    function inner() {
        return num1 * num2;
    }
    return inner();
}

함수안에 함수가 있다. 그래서 내장함수이다. 내부함수 inner는 외부함수 outer의 파라메터 num1 과 지역변수 num2 에 접근이 가능하다는 것만 알아두자. 결과값은 어떻게 알아볼 수 있을까.


var result = outer(7);
console.log(result);		// returns 35 (=7x5)
result = outer(8);
console.log(result);		// returns 40 (=8x5)

outer 함수를 호출할 때마다 num1, num2 의 변수가 생성되고 사용되지만, 함수안의 지역변수이므로 계산값을 리턴하고 나면 더이상 쓸모없는 변수가 되어 둘다 GC(가비지컬렉터)의 수거 대상이 된다. 한마디로 outer(7) 같이 호출하고 난 후에는 num1, num2 변수는 없어진다는 이야기이다. 이것을 유념하고 넘어가자.




이제 Closure 예제를 보자. 내장함수와 같은 맥락이지만 조금 다르다.


function outer(num1) {
    
    function inner(num2) {
        return num1 * num2;
    }
    return inner;
}

위의 내장함수 예제와 모가 다르지.. 라고 생각할 수도 있지만 함수 호출의 결과값을 리턴하는 것이 아니라 함수를 리턴하는 것이 다르다. 내부함수를 리턴하고 있는데 그러면 내부함수에서 참조하는 저 num1 변수의 저장값은 계속 살아있을까 GC에 의해 수거되서 없어지는 값일까. 결론은 내부함수 inner 는 outer 함수의 num1 변수를 참조하고 있기 때문에, 사라지면 안되는 값이 되어버린다. 


실행하는 코드로 명확히 알아보자.


var x = outer(7);
var y = outer(8);

// (...중간로직...)

console.log(x(2));		// returns 14 (=7x2)
console.log(y(3));		// returns 24 (=8x3)


보통 outer(7) 같이 함수를 호출하고 나면, 지역변수들은 변수의 범위를 다하기 때문에 없어져야 하지만, 6~7라인에서 리턴된 함수를 가지고 출력을 할 때 보면 각 리턴된 내부함수에 외부함수의 변수들을 참조하여 계산하고 있는 것을 볼 수 있다.


1. 내부함수의 결과값이 아닌 내부함수 자체를 리턴하고 있다.

2. 그 리턴된 내부함수는 외부함수의 파라메터 및 변수들에 접근가능하며 그 값들을 참조하여 가지고 있다.


라는 게 클로저(Closure)의 핵심이다. 


어려움 없이 이해를 했다면 조금 더 복잡한 Closure 를 알아보자.


var createPet = function(name) {
  var sex;
   
  return {
    setName: function(newName) {
      name = newName;
    },
     
    getName: function() {
      return name;
    },
     
    getSex: function() {
      return sex;
    },
     
    setSex: function(newSex) {
      if(typeof newSex == "string" && 
			(newSex.toLowerCase() == "male" || newSex.toLowerCase() == "female")) {
        sex = newSex;
      }
    }
  }
}
 
var pet = createPet("Vivie");
pet.getName();                  // Vivie
 
pet.setName("Oliver");
pet.setSex("male");
pet.getSex();                   // male
pet.getName();                  // Oliver

createPet 이라는 함수가 있다. 문법이 좀 어려워 보이지만, 단순하게 보면 이 함수는 Object 를 리턴하는 함수일 뿐이다. { } 로 묶여 있는 것을 리턴하기 때문이다. 그 Object 안에는 4가지의 property 가 있는데(setName, getName, getSex, setSex), 모두 함수로 정의되있다. 이것들은 외부함수 변수인 name과 sex를 참조하고 있으며, 이 값들을 변경하고 출력할 수 있는 기능(set, get)을 제공하고 있는 것이다. new 연산자만 없지 사실 java에서 객체를 만들고 메소드를 호출하는 것 처럼 보인다.


출처 : http://emflant.tistory.com/66

'개발 > ' 카테고리의 다른 글

IIFE 와 $(document).ready()  (0) 2017.10.24
javascript garden 번역본  (0) 2017.10.24
spring mvc mocking 사용시 @modelattributes  (0) 2016.12.02
mybatis typealiases  (0) 2016.11.29
spring test mockmvc  (0) 2016.11.29

댓글