Javascript事件处理进阶
这篇文章是我在看乌龟书《编写可维护的Javascript》发现的一篇写的非常好的章节,在这里我并不会教大家什么是绑定事件等比较基础的事。有兴趣了解DOM事件的同学们,可以去w3cschool查阅。
事件绑定
大家都知道前端开发中,事件处理是非常重要的。我们在view层的交互,都是通过绑定事件到UI上,然后我们再处理这些事件。让我们写一个事件绑定的函数先。
var addListener = function(target, type, listener){
if(target.addEventListener){
//2级的DOM事件绑定
target.addEventListener(type, listener, false);
}else if(target.attchEvent){
//IE事件绑定
target.attchEvent("on" + type, listener);
}else{
//0级的DOM事件绑定,也是最可靠的,但只可单次绑定
target["on" + type] = listener;
}
}
上面写了一个兼容性强的DOM事件绑定,但希望大家要注意:DOM0级的事件只可单次绑定,再绑定会导致覆盖的问题出现。
常见用法
当事件触发后,会有一个event回调参数传入事件处理中。而所有的有关事件的信息都会储存在event对象中。我们假设需要做一个点击,并在鼠标的位置弹出框。我们是这样写的。
var handlerClick = function(event){
var popup = document.getElementById("popup");
popup.style.left = event.clientX;
popup.style.top = event.clientY;
popup.className = "active";
}
addListener(element, "click", handlerClick);
乍一看,这是没什么问题的。确实它可以运行的很好,这毋庸置疑。可是你是否有考虑过,你的事件处理程序中,却包含了与用户行为无关的应用逻辑,也就是指弹框这个行为。或许当我们使用mousemove事件时,也需要这段应用逻辑,并可以为此单独拓展一些行为时,我们就只能通过copy同样的应用逻辑并在里面添加自己需要的代码。
而另一个同样明显的缺点,则是关于测试的。如果测试需要通过模拟触发事件的形式进行的,现有的测试框架并不是特别稳定和理想。而将隔离出应用逻辑,可以使我们直接触发功能代码。
规则1:隔离应用逻辑
那我们应该如何拆分应用逻辑和事件处理代码呢?首先我们使用模块模式,将所有有关事件处理的函数放在里面,也能防止全局污染。一起来看看:
var MyAppEvent = {
handlerClick: function(event){
this.showPopup(event);
alert("click it!");
},
handlerMousemove: function(event){
this.showPopup(event);
}
showPopup: function(event){
var popup = document.getElementById("popup");
popup.style.left = event.clientX;
popup.style.top = event.clientY;
popup.className = "active";
}
}
//绑定事件
addListener(element, "click", function(event){
MyAppEvent.handlerClick(event);
})
我们将应用逻辑转移到了showPopup函数中,这样使得弹出框的应用逻辑独立了出来,我们就可以在事件处理上更加灵活。例如我在handlerClick加了一个alert函数,这并不会影响到handlerMousemove里弹框的操作。当然这只是拆解程序代码的第一步。
规则2:不要分发事件对象
我们可以看到在MyAppEvent中,event参数传的到处都是。这其实是没问题的,只是有点无节制的分发罢了。因为我们在showPopup函数中,仅仅只是用到了event对象里的clientX和clientY。其它的都是不必要的。我们应该认为,关于应用逻辑,它其实是不能依赖太多东西的。包括触发的事件(这个我们刚才已经妥善处理了),还有就是回调参数event。
若我们不明确应用逻辑要做什么事情时,直接传入event参数,无可厚非。可是这一次我们其实是明确知道我们仅仅是需要一个x坐标和y坐标,所以我们可以通过事件处理函数对event进行筛选。
var MyAppEvent = {
handlerClick: function(event){
this.showPopup(event.clientX, event.clientY); //改变的地方
alert("click it!");
},
handlerMousemove: function(event){
this.showPopup(event.clientX, event.clientY); //改变的地方
}
showPopup: function(x, y){ //改变的地方
var popup = document.getElementById("popup");
popup.style.left = event.x;
popup.style.top = event.y;
popup.className = "active";
}
}
//绑定事件
addListener(element, "click", function(event){
MyAppEvent.handlerClick(event);
})
那这样写之后,当我们要对这个应用逻辑进行测试时,就不再依赖event对象了,我们可以直接传入应用逻辑所期盼的参数。在这里就是有关x,y的坐标。
// API的参数更加明确,不再依赖event
MyAppEvent.showPopup(10, 10);
当event被提取出来后,应用逻辑则更为独立,不过还有一些小细节要处理。就是关于阻止默认行为event.preventDefault()
与阻止冒泡行为event.stopPropagation()
,则也需要在事件处理函数中提前处理。
最后我们把事件处理程序和应用逻辑之间的分工清晰的分开后,我们可以在很多地方轻松使用相同的业务逻辑,包括前端开发者最头疼的测试代码。
Javascript事件处理进阶的更多相关文章
- 深入理解javascript函数进阶系列第一篇——高阶函数
前面的话 前面的函数系列中介绍了函数的基础用法.从本文开始,将介绍javascript函数进阶系列,本文将详细介绍高阶函数 定义 高阶函数(higher-order function)指操作函数的函数 ...
- javascript入门进阶(一)
javascript 入门进阶 这里主要讲解一下在入门阶段很难注意的一些知识点,不一定有用.但是会了总比不会强. 1.HTML为<script>标签准备的6个属性: -async:可选.表 ...
- javascript事件处理解析
一.什么是事件!(w3c解释) 事件是可以被 JavaScript 侦测到的行为. JavaScript 使我们有能力创建动态页面.事件是可以被 JavaScript 侦测到的行为. 网页中的每个元素 ...
- JavaScript事件处理的三种方式(转)
一.什么是JavaScript事件? 事件(Event)是JavaScript应用跳动的心脏,也是把所有东西粘在一起的胶水,当我们与浏览器中Web页面进行某些类型的交互时,事件就发生了. 事件可能是用 ...
- 使用AmplifyJS和JQuery编写更好更优雅的javascript事件处理代码
事件(或消息)是一种经常使用的软件设计模式.可以减少消息处理者和消息公布者的之间的耦合,比方J2EE里面的JMS规范.设计模式中的观察者模式(也叫公布/订阅模式).这对于javascript代码相同适 ...
- 私人定制javascript事件处理机制(浅谈)
看到园子里关于事件监听发表的文章,我都有点不好意思写了.不过想想我的题目以私人定制作开头也就妥妥地写吧. 事件相关概念 1.事件类型 发生事件的字符串 有传统事件类型 比如表单.window事件等 D ...
- JavaScript的进阶之路(七)客户端JavaScript知识点总结
一.客户端JavaScript主要是BOM DOM的操作和js脚本的兼容性.互用性.可访问性.安全性的应用.以及一些框架的引用. 二.BOM:浏览器对象模型 主要介绍window对象 1.定时器:se ...
- javascript——事件处理模型(DOM 和 IE)
javascript的事件处理模型分为 DOM事件处理模型和 IE事件处理模型. 一.DOM事件流模型 DOM事件流分为三个阶段:捕获阶段.目标阶段.冒泡阶段. 捕获阶段:自上而下,由document ...
- 【WIP】客户端JavaScript 事件处理
创建: 2017/10/15 完成: 2017/10/15 更新: 2017/11/04 加粗事件的参数 更新: 2017/12/12 增加事件处理时获取事件对象的方法 更新: 2019/05/2 ...
随机推荐
- URAL1057. Amount of Degrees(DP)
1057 简单的数位DP 刚开始全以2进制来算的 后来发现要找最接近x,y值的那个基于b进制的0,1组合 #include <iostream> #include<cstdio&g ...
- 编程时 对 用途这个字段定义时 不要用using 这个英文
编程时 对 用途这个字段定义时 不要用using 这个英文
- Spring 实践 -IoC
Spring 实践 标签: Java与设计模式 Spring简介 Spring是分层的JavaSE/EE Full-Stack轻量级开源框架.以IoC(Inverse of Control 控制反转) ...
- Matlab中plot函数参数解析
功能 二维曲线绘图 语法 plot(Y) plot(X1,Y1,...) plot(X1,Y1,LineSpec,...) plot(...,'PropertyName',PropertyValue, ...
- scala学习笔记(5)
偏应用函数 举个例子 def sum(a: Int, b: Int, c: Int) = a + b + c val a = sum _ println(a(1,2,3)) 实际发生的事情是这样的:名 ...
- openssl rsa 加解密
<h4>1.openssl进行rsa加密解密</h4>首先介绍下命令台下openssl工具的简单使用:生成一个密钥:<pre lang="c" esc ...
- SQL利用Case When Then多条件判断
CASE WHEN 条件1 THEN 结果1 WHEN 条件2 THEN 结果2 WHEN 条件3 THEN 结果3 WHEN 条件4 THEN 结果4 ....... ...
- [转]vi与vim的区别
一直用着vi,有朋友劝我用vim,那么它们有什么区别呢? 简单点来说,它们都是多模式编辑器, 不同的是vim 是vi的升级版本,它不仅兼容vi的所有指令, 而且还有一些新的特性在里面. vim的这些优 ...
- swfupload 参数说明
一.配置参数对象中的常用属性及说明 属性 类型 默认值 描述 upload_url String 处理上传文件的服务器端页面的url地址,可以是绝对地址,也可以是相对地址,当为相对地址时相对的是当 ...
- ylb:表的结构的修改和基本约束
ylbtech-SQL Server:SQL Server-表的结构的修改和基本约束 SQL Server 表的结构的修改和基本约束. 1,表的结构的修改和基本约束返回顶部 use master go ...