事件及事件流
题目1:DOM0 事件和DOM2级在事件监听使用方式上有什么区别?
DOM0 事件:
- HTML内联方式:
<div onclick="alert()"></div>
缺点:
-
存在加载顺序问题,如果事件处理程序在html代码之后加载,用户
可能在事件处理程序还未加载完成时就点击按钮之类的触发事件,存在时间差问题 -
这样书写html代码和JavaScript代码紧密耦合,维护不方便
- JavaScript指定事件处理程序:通过JavaScript指定事件处理程序就是把一个方法赋值给一个元素的事件处理程序属性。
<input id="btnClick" type="button" value="Click Here" />
<script type="text/javascript">
var btnClick = document.getElementById('btnClick');
btnClick.onclick = function showMessage() {
alert(this.id);
};
</script>
事件处理程序被认为是元素的方法,事件处理程序在元素的作用域下运行,this就是当前元素。当我们删除事件处理程序的时候,只需把元素的onclick属性赋为null即可。
缺点:同一个事件只能定义一个监听函数,也就是说,如果定义两次onclick属性,后一次定义会覆盖前一次
DOM2级:两个方法用于处理指定和删除事件处理程序的操作:
- addEventListener
- removeEventListener
这两个方法都接受三个参数;
- type:事件名称,大小写敏感。
- listener:监听函数。事件发生时,会调用该监听函数。
- useCapture:布尔值,表示监听函数是否在捕获阶段(capture)触发
<input id="btnClick" type="button" value="Click Here" />
<script type="text/javascript">
var btnClick = document.getElementById('btnClick');
btnClick.addEventListener('click', function() {
alert(this.id);
}, false);
</script>
通过addEventListener添加的事件处理程序只能通过removeEventListener移除,移除时参数与添加的时候相同,这就意味着刚才我们添加的匿名函数无法移除,因为匿名函数虽然方法体一样,但是句柄却不相同,所以当我们有移除事件处理程序的时候可以这样写。
input id="btnClick" type="button" value="Click Here" />
<script type="text/javascript">
var btnClick = document.getElementById('btnClick');
var handler=function() {
alert(this.id);
}
btnClick.addEventListener('click', handler, false);
btnClick.removeEventListener('click', handler, false);
</script>
优点:
- 可以针对同一个事件,添加多个监听函数。
- 能够指定在哪个阶段(捕获阶段还是冒泡阶段)触发回监听函数。
- 除了DOM节点,还可以部署在window、XMLHttpRequest等对象上面,等于统一了整个JavaScript的监听函数接口。
this是一个让人很头疼的问题,这里补充一下:
- 以下写法的this对象都指向Element节点
// JavaScript代码
element.onclick = print
element.addEventListener('click', print, false)
element.onclick = function () {console.log(this.id);}
// HTML代码
<element onclick="console.log(this.id)">
2.以下写法的this对象,都指向全局对象。
// JavaScript代码
element.onclick = function (){ doSomething() };
element.setAttribute('onclick', 'doSomething()');
// HTML代码
<element onclick="doSomething()">
题目2: attachEvent与addEventListener的区别?
-
参数个数不相同,这个最直观,addEventListener有三个参数,attachEvent只有两个,attachEvent添加的事件处理程序只能发生在冒泡阶段,addEventListener第三个参数可以决定添加的事件处理程序是在捕获阶段还是冒泡阶段处理
-
第一个参数意义不同,addEventListener第一个参数是事件类型(比如click),而attachEvent第一个参数指明的是事件处理函数名称(onclick)
-
事件处理程序的作用域不相同,addEventListener的作用域是元素本身,this是指的触发元素,而attachEvent事件处理程序会在全局变量内运行,this是window
-
为一个事件添加多个事件处理程序时,执行顺序不同,addEventListener添加会按照添加顺序执行,而attachEvent添加多个事件处理程序时顺序无规律(添加的方法少的时候大多是按添加顺序的反顺序执行的,但是添加的多了就无规律了),所以添加多个的时候,不依赖执行顺序的还好,若是依赖于函数执行顺序,最好自己处理,不要指望浏览器。
题目3: 解释IE事件冒泡和DOM2事件传播机制?
IE事件冒泡:从目标节点依次传导到window对象,自下而上,由于像水冒泡,因此称为冒泡事件。
DOM2事件流:
第一阶段:从window对象传导到目标节点,称为“捕获阶段”
第二阶段:在目标节点上触发,称为“目标阶段”
第三阶段:从目标节点传导回window对象,称为“冒泡阶段”
这种三阶段的传播模型,会使得一个事件在多个节点上触发。
题目4:如何阻止事件冒泡? 如何阻止默认事件?
阻止事件冒泡:
- IE:
e.cancelBubble = true;
- 非IE:
e.stopPropagation();
阻止默认事件:
- IE:
e.returnValue = false;
- 非IE:
e.preventDefault();
题目5:有如下代码,要求当点击每一个元素li时控制台展示该元素的文本内容。不考虑兼容
<ul>
<li>1</li>
<li>2</li>
<li>3</li>
</ul>
<script>
var $ = (className) => document.querySelector(className)
$('ul').addEventListener('click',function(e){
var target = e.target
if(target.tagName.toLowerCase() === 'li'){
console.log(target.textContent)
}
})
</script>
题目6: 补全代码,要求:
当点击按钮开头添加时在<li>这里是</li>元素前添加一个新元素,内容为用户输入的非空字符串;当点击结尾添加时在最后一个 li 元素后添加用户输入的非空字符串.
当点击每一个元素li时控制台展示该元素的文本内容。
<ul class="ct">
<li>这里是</li>
<li>饥人谷</li>
<li>任务班</li>
</ul>
<input class="ipt-add-content" placeholder="添加内容"/>
<button id="btn-add-start">开头添加</button>
<button id="btn-add-end">结尾添加</button>
<script>
var $ = (className) => document.querySelector(className)
$('#btn-add-start').addEventListener('click',function(){
if($('.ipt-add-content').value === ''){
alert('请输入一些内容吧');
return;
}
var newLi = document.createElement('li')
newLi.textContent = $('.ipt-add-content').value
$('.ct').insertBefore(newLi,$('.ct').firstChild)
})
$('#btn-add-end').addEventListener('click',function(){
if($('.ipt-add-content').value === ''){
alert('请输入一些内容吧');
return;
}
var newLi = document.createElement('li')
newLi.textContent = $('.ipt-add-content').value
$('.ct').appendChild(newLi)
})
$('ul').addEventListener('click',function(e){
var target = e.target
if(target.tagName.toLowerCase() === 'li'){
console.log(target.textContent)
}
})
</script>
可查看旁边的demo:demo
题目7: 补全代码,要求:当鼠标放置在li元素上,会在img-preview里展示当前li元素的data-img对应的图片。
<ul class="ct">
<li data-img="">鼠标放置查看图片1</li>
<li data-img="http://tse2.mm.bing.net/th?id=OIP.pRu5ddfv1MQlcexX9cOCuwDlEs&w=198&h=259&c=7&qlt=90&o=4&dpr=1.5&pid=1.7">鼠标放置查看图片2</li>
<li data-img="http://tse3.mm.bing.net/th?id=OIP.eWHsr9TKLGWTYjBDAx-qMgDIEs&w=198&h=297&c=7&qlt=90&o=4&dpr=1.5&pid=1.7">鼠标放置查看图片3</li>
</ul>
<div class="img-preview"></div>
<script>
var $ = (className) => document.querySelector(className)
var $$ = (className) => document.querySelectorAll(className)
var lis = $$(".ct li");
var imgPre = $(".img-preview");
function createImg() {
var img = new Image;
img.setAttribute("src", this.dataset.img);
imgPre.appendChild(img);
}
for(var i=0; i<lis.length; i++) {
lis[i].addEventListener("mouseover",createImg);
lis[i].addEventListener("mouseout", function() {
imgPre.removeChild(imgPre.querySelector("img"));
})
}
</script>
可查看旁边的demo:demo