함수는 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();
클로저를 사용하여 코드 최적화:
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); * } */
var oEvent = {}; (function(){ var addEvent = function(){ /*代码的实现省略了*/ }; function removeEvent(){} oEvent.addEvent = addEvent; oEvent.removeEvent = removeEvent; })();
이 코드를 사용하고 싶습니다:
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);
var outer = null; (function(){ var one = 1; function inner (){ one += 1; alert(one); } outer = inner; })(); outer(); //2 outer(); //3 outer(); //4
주목
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); }; }
你会发现当鼠标移过每一个
解决方法一:
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 内存泄露
使用闭包十分容易造成浏览器的内存泄露,严重情况下会是浏览器挂死