>웹 프론트엔드 >JS 튜토리얼 >js closure_javascript 기술로 인한 이벤트 등록 문제 분석

js closure_javascript 기술로 인한 이벤트 등록 문제 분석

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

배경: 여가 시간에 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"가 출력됩니다

위 내용은 이 글의 전체 내용입니다. 모든 분들의 공부에 도움이 되었으면 좋겠습니다.

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