javaScript hook
今天在网上搜索了不少资料,基本概念如下:
钩子(Hook),是Windows消息处理机制的一个平台,应用程序可以在上面设置子程以监视指定窗口的某种消息,而且所监视的窗口可以是其他进程所创建的。当消息到达后,在目标窗口处理函数之前处理它。钩子机制允许应用程序截获处理window消息或特定事件。
钩子实际上是一个处理消息的程序段,通过系统调用,把它挂入系统。每当特定的消息发出,在没有到达目的窗口前,钩子程序就先捕获该消息,亦即钩子函数先得到控制权。这时钩子函数即可以加工处理(改变)该消息,也可以不作处理而继续传递该消息,还可以强制结束消息的传递。
转自:http://www.clanfei.com/2013/10/1730.html
当我们面对比较复杂的前端项目时,我们经常会采用模块化的方式来对JavaScript代码进行解耦,以方便对代码的管理和维护,以下是一个简单的立即执行函数实现的模块化例子:
var Common = (function(){
//这个函数是立即执行的,执行结果给了Common
var func = function(){
// 全局公用方法
}
return {
func : func
}
})();
var ModuleA = (function(){
var _count = 1;
var init = function(){
// 独立模块逻辑
}
var getCount = function(){
return _count;
}
return {
init : init,
getCount : getCount
}
})();
模块只对外暴露外部需要的接口,而外部模块不需要关心其内部的运行逻辑,只需要知道调用接口的方式和返回结果,这样就实现了模块的“低耦合,高内聚”。
看起来很美好,可是当项目的逻辑变的越来越复杂时,比如A模块中某个针对全局公用的逻辑,可能在B模块的某种特定情况下需要进行一些额外的逻辑操作,该怎么办呢?
var Common = (function(){
var func = function(){
// 全局公用方法
if(typeof ModuleA != 'undefined' && ModuleA.getCount() > 0){
// 模块A需要进行的额外逻辑操作,实际上就是钩子捕获消息
}
}
return {
func : func
}
})();
var ModuleA = (function(){
var _count = 1;
var init = function(){
// 独立模块逻辑
}
var getCount = function(){
return _count;
}
return {
init : init,
getCount : getCount
}
})();
不知道当你看到Common.func中间的那一坨东西的时候,会不会突然怒吼一声:“卧槽,尼玛怎么这么恶心!”= =。。
明明是A模块的逻辑,却恶心地出现在了公用模块里,如果这样的特殊逻辑多起来之后,Common模块会不会变成这样?
var Common = (function(){
var func = function(){
// 全局公用方法
if(typeof ModuleA != 'undefined' && ModuleA.getCount() > 0){
// 模块A需要进行的额外逻辑操作
}
if(typeof ModuleB != 'undefined' && ModuleB.getWhat() != 'something'){
// 模块B需要进行的额外逻辑操作
}
// ...
if(typeof ModuleZ != 'undefined' && ModuleB.isWhat() !== false){
// 模块Z需要进行的额外逻辑操作
}
}
return {
func : func
}
})();
天哪,简直无法忍受。。
如果。。如果有这么一个钩子(Hook),可以把额外的逻辑代码挂在Common.func上,而Common.func执行的时候顺便把钩子上挂着的代码也执行了,那该多好啊。。这样的话既可以实现特殊的额外操作,又可以保持模块的低耦合和高内聚:
var Common = (function(){
//这个函数是立即执行的,执行结果给了Common
var func = function(){
//执行挂在这个方法的钩子上的所有额外逻辑代码
var arg = 2;
Hook.doActions(arg);//每个模块都可以得到arg,根据它做相应操作。但是并不能更改arg;
console.log(arg);
//全局公用方法
}
return {
func: func
}
})()
//应用场景:比如A模块中某个针对全局公用的逻辑,可能在B模块的某种特定情况下需要进行一些额外的逻辑操作。
var ModuleA = (function(){
var _count = 1;
var init = function(){
//用钩子把额外的逻辑挂到Common.func上
Hook.addAction(Common.func, function(arg){
if(_count > 0){
//增加的额外逻辑操作
arg = arg + 1;
console.log("看看执行到没?");
console.log(arg);
}
})
}
var getCount = function(){
return _count;
}
return {
init: init,
getCount: getCount
}
})()
ModuleA.init();
Common.func();
没有不可能。借鉴了一下WordPress的Hook机制,一个基于JavaScript钩子机制模块就实现了。
当然,一个完整的钩子机制需要考虑的并不像上面说的那么简单,具体的实现大家请看代码,或者懒得看的可以自己尝试实现,我就不在赘述了:
<script>
/*var fn1 = {
_hooks : {
actions:[
{action:action,priority:priority},
{}
],
filters:[
{
filter: filter,
priortity: priority
}
]
}
}*/
var Hook = (function(){
var addAction = function(method, action, priority){
_initHook(method);
var actions = method['_hooks_'].actions;
actions.push({
action: action,
priority: priority || 10,
})
actions.sort(_compare);
}
var doActions = function(){
var method = Hook.doActions.caller;
//caller:http://www.cnblogs.com/darr/p/4756906.html
console.log(arguments.callee);
//callee;
//Common.func
//指向调用它的函数
//如果一个函数f是在全局作用域内被调用的,则f.caller为null,相反,如果一个函数是在另外一个函数作用域内被调用的,则f.caller指向调用它的那个函数.
_initHook(method);
//Common.func._hooks_:{actions:[],filters:[]}
var actions = method['_hooks_'].actions;
if(arguments.length == 0){
arguments = method.arguments;//?
}
for(var i in actions){
if(actions[i].action.apply(method, arguments) === false){
//apply:http://www.cnblogs.com/darr/p/4757082.html
return false;
}
}
}
var hasAction = function(method, action){
_initHook(method);
var actions = method['_hooks_'].actions;
if(actions.length > 0 && action != undefined){
for(var i in actions){
if(actions[i].action == action){
return true;
}
}
return false;
}else{
return actions.lengths > 0;
}
}
var removeAction = function(method, action){
_initHook(method);
var actions = method['_hooks_'].actions;
if(actions.length > 0){
if(actions != undefined){
for(var i in actions){
if(actions[i].action == action){
delete actions[i];
return;
}
}
}else{
method['_hooks_'].actions = [];
}
}
}
var addFilter = function(method, filter, priority){
_initHook(method);
var filters = method['_hooks_'].filters;
filters.push({
filter:filter,
priority: priority || 10
});
filters.sort(_compare)
}
var applyFilters = function(value){
var method = Hook.applyFilters.caller;
_initHook(method);
var filters = method['_hooks_'].filters;
for(var i in filters){
value = filters[i].filter.call(method, value);
}
return value;
}
var hasFilter = function(method, filter){
_initHook(method);
var filters = method['_hooks_'].filters;
if(filters.length > 0 && filter != undefined){
for(var i in filters){
if(filters[i] == filter){
return true;
}
}
return false;
}else{
return filters.length > 0 ;
}
}
var removeFilter = function(method, filter){
_initHook(method);
var filters = method['_hooks_'].filters;
if(filters.length > 0){
if(filter != undefined){
for(var i in filters){
if(filters[i] == filter){
delete filters[i];
return;
}
}
}else{
//filters = [];
//有坑啊,不能用filters。应该将method['_hooks_'].filters指向空地址
method['_hooks_'].filters = [];
}
}
}
var _compare = function(hook1, hook2){
return hook1.priority < hook2.priority;
}
var _initHook = function(method){
if(!method['_hooks_']){
method['_hooks_'] = {
actions: [],
filters: []
}
}
}
return {
addAction: addAction,
doActions: doActions,
hasAction: hasAction,
removeAction: removeAction,
addFilter: addFilter,
applyFilters: applyFilters,
hasFilter: hasFilter,
removewFilter: removeFilter
}
})();
</script>
像适配器模式:http://www.cnblogs.com/tomxu/archive/2012/04/11/2435452.html
像观察者模式:http://www.cnblogs.com/TomXu/archive/2012/03/02/2355128.html
javaScript hook的更多相关文章
- XSS攻击:获取浏览器记住的明文密码
作者:余弦(@evilcos) 0x01. XSS获取明文密码的多种方式 我已经感受到Web潮流带来的巨大革新,尤其是最近HTML5越来越火.浏览器们在客户端瓜分着这个Web OS,只要是对用户体验好 ...
- CSS架构目标
擅长CSS的Web开发人员不仅可以从视觉上复制实物原型,还可以用代码进行完美的呈现.无需使用表格.尽可能少的使用图片.如果你是个名副其实的高手,你可以快速把最新和最伟大的技术应用到你的项目中,比如媒体 ...
- css best practice for big team and project
推荐查看以下文章: https://segmentfault.com/a/1190000000704006 关于BEM,SMACSS,OOCSS的通俗易懂的介绍 http://philipwalton ...
- [hook.js]通用Javascript函数钩子及其他
2013.02.16<:article id=post_content> 最近看Dom Xss检测相关的Paper,涉及到Hook Javascript函数,网上翻了一下,貌似没有什么通用 ...
- JavaScript常用的Hook脚本
JavaScript常用的Hook脚本 本文Hook脚本 来自 包子 页面最早加载代码Hook时机 在source里 用dom事件断点的script断点 然后刷新网页,就会断在第一个js标签,这时候就 ...
- 闭包传参 余额计算 钩子hook 闭包中的this JavaScript 钩子
闭包传参 余额计算 钩子hook 小程序 a=function(e){console.log(this)}() a=function(e){console.log(this)}() VM289 ...
- javascript单元测试框架mochajs详解
关于单元测试的想法 对于一些比较重要的项目,每次更新代码之后总是要自己测好久,担心一旦上线出了问题影响的服务太多,此时就希望能有一个比较规范的测试流程.在github上看到牛逼的javascript开 ...
- JavaScript学习笔记(四)——jQuery插件开发与发布
jQuery插件就是以jQuery库为基础衍生出来的库,jQuery插件的好处是封装功能,提高了代码的复用性,加快了开发速度,现在网络上开源的jQuery插件非常多,随着版本的不停迭代越来越稳定好用, ...
- JavaScript学习总结(四)——jQuery插件开发与发布
jQuery插件就是以jQuery库为基础衍生出来的库,jQuery插件的好处是封装功能,提高了代码的复用性,加快了开发速度,现在网络上开源的jQuery插件非常多,随着版本的不停迭代越来越稳定好用, ...
随机推荐
- Linux服务器偶尔无法访问问题
最近上了一台web服务器(本地包含mysql服务器),在运行一段时间发现服务器偶尔会无法访问, 包括mysql,ftp以及ssh等都无法响应,但是已经连接上的ssh不受任何影响,在查看系统log时, ...
- type tips
网上有这么一篇文章,全文如下:http://bbs.9ria.com/blog-220191-18429.html AS3中一共有以下六种获取变量类型的方法: typeof instanceof ...
- 三:分布式事务一致性协议2pc和3pc
一:分布式一致性协议--->对于一个分布式系统进行架构设计的过程中,往往会在系统的可用性和数据一致性之间进行反复的权衡,于是就产生了一系列的一致性协议.--->长期探索涌现出一大批经典的一 ...
- RequireJS 2.0 正式发布(转)
RequireJS发布了一个大版本,直接从version1.0.8升级到了2.0.随后的几小时James Burke又迅速的将版本调整为2.0.1,当然其配套的打包压缩工具r.js也同时升级到了2.0 ...
- js+css实现带缓冲效果右键弹出菜单
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- window 7 下一台cp 两个mysql 配置主从
环境 : 个人 pc windows7 一台 ; 使用 : 官方下载: mysql-noinstall-5.5.11-win32.zip 1. 解压 成2个 (文件夹) mysql_master (主 ...
- YII内置验证规则
required: 必填字段验证, 来自 CRequiredValidator类的别名 array(‘字段名列表用逗号隔开’, ‘required’), 就这样的一个小小的写法,可以让字段前面加 ...
- jQuery 关于点击菜单项,使子条目“向上”展开效果的实现
为什么做了这样一个的功能呢?原因是前一段时间jQuery群里有个朋友想实现这样一个东东,大家都知道jQuery中有现成的slideDown和slideUp方法,但那是向下展开,而这个是一个完全相反的效 ...
- python在linux上的GUI无法弹出界面
在进行python写GUI程序的时候,使用Tkinter,发现无法执行程序,报错如下: X connection to localhost:10.0 broken(explicit kill or s ...
- 用js完成blog项目
//前台调用 var $ = function (args) { return new Base(args); } //基础库 function Base(args) { //创建一个数组,来保存获取 ...