배경: 여가 시간에 js 스코프 체인 및 클로저에 관한 몇 가지 기사를 읽었다가 우연히 이전에 직면했던 문제를 보았습니다. 이는 for 루프 등록 이벤트에 있습니다. dom 노드용 드라이버, 자세한 내용은 아래 코드를 참조하세요.
<!DOCTYPE html> <html> <head> <title>js闭包</title> <meta charset="utf-8" /> </head> <body> <button id="anchor1">1</button> <button id="anchor2">2</button> <button id="anchor3">3</button> <script type="text/javascript" src="jquery-1.12.1.js"></script> <script type="text/javascript"> function pageLoad(){ for (var i = 1; i <=3; i++) { var anchor = document.getElementById("anchor" + i); anchor.onclick = function () { console.log("anchor"+i); } } } window.onload = pageLoad; </script> </body> </html>
일반적인 생각에 따르면 3개의 버튼을 클릭하면 각각 "anchor1", "anchor2" 및 "anchor3"이 표시되는 결과가 나와야 합니다. 기간 초반에는 그렇게 생각했지만 결과는 무엇이든 마찬가지입니다. 버튼을 클릭하면 "anchor4"라는 메시지가 표시됩니다.
이게 왜요? 걱정하지 마세요. js 스코프 체인과 클로저에 대한 지식이 포함되어 있으므로 여기서는 자세히 소개하지 않겠습니다.
먼저 Anchor.onclick이 무엇인지 살펴보겠습니다. 이건 DOM 레벨 0 이벤트 핸들러입니다. 말도 안 돼요. 블로거는 사이코패스인가요? *************** 제가 하고 싶은 말은 이 앵커입니다. 🎜 >
은 var name="Xiao Ming"과 같은 이벤트 핸들러의 선언입니다. 이것이 선언되었지만 아직 실행되지 않았습니다. 이것이 핵심입니다. 위의 js 코드를 살펴보겠습니다. 🎜>
function pageLoad(){ for (var i = 1; i <=3; i++) { var anchor = document.getElementById("anchor" + i); anchor.onclick = function () { console.log("anchor"+i); } if(i==2){ debugger;//我们在这里debugger一下,然后在控制台手动触发#anchor1和#anchor2的点击事件 } } } window.onload = pageLoad;
i==2일 때 디버거를 사용하여 루프를 중지한 다음 콘솔로 이동하여 #anchor1 및 #anchor2의 클릭 이벤트를 수동으로 트리거하면 콘솔에 "anchor2"가 인쇄됩니다.
전체 논리는 대략 다음과 같습니다. Anchor.onclick은 항상 i의 참조를 저장하고 i는 루프 중에 i=1에서 i=4로 계속 변경됩니다. 하지만 루프 중에 anchor.onclick은 한 번 저장됩니다. "한 번"이라는 단어),
1, 2, 3의 세 가지 경우가 있는데 i는 결국 4가 되기 때문에 어떤 버튼을 눌러도 "anchor4"가 출력됩니다
결론: js에서 스코프 체인과 클로저에 대한 지식은 매우 중요합니다. 여기서는 언급하지 않았지만 실제로는 명확하게 설명되지 않을 것 같습니다. 그러면 모두가 오해하게 될 것입니다
JS 클로저로 인한 이벤트 등록 문제에 대한 위 소개는 모두 편집자가 공유한 내용이므로 참고가 되기를 바라며, Script Home에 많은 지원 부탁드립니다.