>  기사  >  웹 프론트엔드  >  Javascript 익명 함수 및 자체 실행 함수에 대한 간략한 분석_javascript 기술

Javascript 익명 함수 및 자체 실행 함수에 대한 간략한 분석_javascript 기술

WBOY
WBOY원래의
2016-05-16 15:16:001055검색

함수는 JavaScript에서 가장 유연한 객체입니다. 여기서는 익명 함수의 사용에 대해서만 설명합니다. 익명 함수: 함수 이름이 없는 함수입니다.

함수의 정의는 크게 세 가지로 나눌 수 있습니다.

첫 번째 유형: 가장 일반적인 유형이기도 합니다

function double(x){ 
return 2 * x; 
}

두 번째 방법: 이 방법은 함수 생성자를 사용하고 매개변수 목록과 함수 본문을 모두 문자열로 처리하므로 매우 불편하므로 권장되지 않습니다.

var double = new Function('x', 'return 2 * x;');

세 번째 유형:

var double = function(x) { return 2* x; }

"=" 오른쪽에 있는 함수는 익명 함수라는 점에 유의하세요. 함수를 생성한 후 해당 함수는 square 변수에 할당됩니다.

익명함수 생성

첫 번째 방법은 위에서 언급한 것처럼 제곱함수를 정의하는 방법인데, 이 역시 가장 많이 사용되는 방법 중 하나입니다.

두 번째 방법:

(function(x, y){ 
alert(x + y); 
})(2, 3);

여기서(첫 번째 대괄호 안에) 익명 함수가 생성되고, 두 번째 대괄호는 익명 함수를 호출하고 매개변수를 전달하는 데 사용됩니다. 괄호는 표현식이고 표현식에는 반환값이 있으므로 뒤에 괄호 한 쌍을 추가하여 실행할 수 있습니다.

자체 실행 익명 함수

1. 자체 실행 익명 함수란 무엇인가요?

다음과 같은 함수를 말합니다: (함수 {//코드})();

2. 질문

(function {// code})();를 실행할 수 있지만 {// code}() 함수가 오류를 보고하는 이유는 무엇인가요?

3. 분석

(1) 우선 둘의 차이점을 이해해야 합니다.
(함수 {//코드})는 표현식이고, 함수 {//코드}는 함수 선언입니다.
(2) 둘째, js "사전 컴파일"의 특징:
js의 "컴파일 전" 단계에서는 함수 선언이 해석되지만 표현식은 무시됩니다.
(3) js가 function() {//code}();를 실행할 때 function() {//code}는 "사전 컴파일" 단계에서 해석되었으므로 js는 function(){//code}을 건너뜁니다. ();을 실행하려고 하면 오류가 보고됩니다.
js가 (function {// code})();를 실행할 때 (function {// code})는 표현식이므로 js는 이를 해결하여 반환 값을 가져옵니다. 따라서 ()를 만나게 됩니다. ;, 실행됩니다.

또한 함수를 표현식으로 변환하는 방법이 반드시 그룹화 연산자()에 의존하는 것은 아닙니다. void 연산자, ~ 연산자, !

예:

!function(){ 
alert("另类的匿名函数自执行"); 
}();

익명 함수 및 클로저

클로저(Closure)의 영어 단어는 클로저(closure)인데, 이는 자바스크립트 지식에서 매우 중요한 부분입니다. 왜냐하면 클로저를 사용하면 코드의 양을 크게 줄이고, 코드를 더 명확하게 보이게 만드는 등의 작업을 할 수 있기 때문입니다. 한마디로 매우 강력합니다.

클로저의 의미: 직설적으로 말하면 클로저는 함수의 중첩입니다. 외부 함수가 실행되더라도 내부 함수는 외부 함수의 모든 변수를 사용할 수 있습니다(여기에는 JavaScript 범위 체인이 포함됩니다).

function checkClosure(){ 
var str = 'rain-man'; 
setTimeout( 
function(){ alert(str); } //这是一个匿名函数 
, 2000); 
} 
checkClosure();

이 예제는 실행 프로세스를 주의 깊게 분석한 후에도 여전히 많은 지식 포인트가 있습니다. checkClosure 함수의 실행은 즉각적이며(아마도 0.00001밀리초만 소요될 수 있음) 변수 str이 함수 본문에 생성됩니다. checkClosure.의 경우 setTimeout의 익명 함수에 str에 대한 참조가 있으므로 checkClosure가 실행된 후에 str이 해제되지 않습니다. 2초 후 함수 본문의 익명 함수가 실행되고 str이 해제됩니다.

클로저를 사용하여 코드 최적화:

function forTimeout(x, y){ 
alert(x + y); 
} 
function delay(x , y , time){ 
setTimeout('forTimeout(' + x + ',' + y + ')' , time); 
} 
/** 
* 上面的delay函数十分难以阅读,也不容易编写,但如果使用闭包就可以让代码更加清晰 
* function delay(x , y , time){ 
* setTimeout( 
* function(){ 
* forTimeout(x , y) 
* } 
* , time); 
* } 
*/
익명 함수의 가장 큰 용도는 클로저(JavaScript 언어의 기능 중 하나)를 만드는 것이고, 네임스페이스를 구축하여 전역 변수의 사용을 줄일 수도 있습니다.

var oEvent = {}; 
(function(){ 
var addEvent = function(){ /*代码的实现省略了*/ }; 
function removeEvent(){} 

oEvent.addEvent = addEvent; 
oEvent.removeEvent = removeEvent; 
})();
이 코드에서 addEvent, RemoveEvent 함수는 로컬 변수이지만, 전역 변수 oEvent를 통해 사용할 수 있어 전역 변수의 사용이 대폭 줄어들고 웹 페이지의 보안이 강화됩니다.

이 코드를 사용하고 싶습니다:

oEvent.addEvent(document.getElementById('box') , 'click' , function(){});
var rainman = (function(x , y){ 
return x + y; 
})(2 , 3); 
/** 
* 也可以写成下面的形式,因为第一个括号只是帮助我们阅读,但是不推荐使用下面这种书写格式。 
* var rainman = function(x , y){ 
* return x + y; 
* }(2 , 3);
여기에서는 변수 rainman을 만들고 익명 함수를 직접 호출하여 5로 초기화합니다. 이 작은 트릭은 때때로 매우 실용적입니다.

var outer = null; 
(function(){ 
var one = 1; 
function inner (){ 
one += 1; 
alert(one); 
} 
outer = inner; 
})(); 
outer(); //2 
outer(); //3 
outer(); //4
이 코드의 변수 one은 지역 변수이므로(함수 내에서 정의되기 때문에) 외부에서 접근할 수 없습니다. 하지만 여기서는 변수 1에 액세스할 수 있는 내부 함수를 만들었고 전역 변수인 외부는 내부를 참조하므로 외부를 세 번 호출하면 증분 결과가 나타납니다.

주목

1 클로저를 사용하면 내부 함수가 상위 함수의 변수를 참조할 수 있지만 변수는 최종 값입니다

/** 
* <body> 
* <ul> 
* <li>one</li> 
* <li>two</li> 
* <li>three</li> 
* <li>one</li> 
* </ul> 
*/ 
var lists = document.getElementsByTagName('li'); 
for(var i = 0 , len = lists.length ; i < len ; i++){ 
lists[ i ].onmouseover = function(){ 
alert(i); 
}; 
}

你会发现当鼠标移过每一个25edfb22a4f469ecb59f1190150159c6元素时,总是弹出4,而不是我们期待的元素下标。这是为什么呢?注意事项里已经讲了(最终值)。显然这种解释过于简单,当mouseover事件调用监听函数时,首先在匿名函数( function(){ alert(i); })内部查找是否定义了 i,结果是没有定义;因此它会向上查找,查找结果是已经定义了,并且i的值是4(循环后的i值);所以,最终每次弹出的都是4。

解决方法一:

var lists = document.getElementsByTagName('li'); 
for(var i = 0 , len = lists.length ; i < len ; i++){ 
(function(index){ 
lists[ index ].onmouseover = function(){ 
alert(index); 
}; 
})(i); 
}

解决方法二:

var lists = document.getElementsByTagName('li'); 
for(var i = 0, len = lists.length; i < len; i++){ 
lists[ i ].$$index = i; //通过在Dom元素上绑定$$index属性记录下标 
lists[ i ].onmouseover = function(){ 
alert(this.$$index); 
}; 
}

解决方法三:

function eventListener(list, index){ 
list.onmouseover = function(){ 
alert(index); 
}; 
} 
var lists = document.getElementsByTagName('li'); 
for(var i = 0 , len = lists.length ; i < len ; i++){ 
eventListener(lists[ i ] , i); 
}

2 内存泄露

使用闭包十分容易造成浏览器的内存泄露,严重情况下会是浏览器挂死

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.