Javascript 判断函数类型完美解决方案


Posted in Javascript onSeptember 02, 2009

但是,有一些细节并不为我们所熟知。John Resig 在分析了这些细节之后,为我们提供了一个完美的解决方案,本文将作详细介绍:

一、传统方法不为人所知的细节
毫无疑问,在判断函数类型时,我们使用的是typeof方法,比如:

function fn(){ 
//content 
} 
alert(typeof fn)//结果是"function"。

但是,该方法在一些浏览器中并不是像我们想像的那样工作。

1、Firefox2和Firefox3
在这两个浏览器中,用typeof检测HTML对象元素的类型,得到是一个不精确的“function”结果,而不是“object”,如HTMLDocument。如:

alert(typeof HTMLDocument); 
//在Firefox2中结果是"function"; 
//在Firefox3中结果是"object";

2、Firefox2
对于正则表达式,在该浏览器中返回的结果是“function”(在Firefox3中结果是“object”),如:
var reg = /test/; 
alert(typeof reg); 
//在Firefox2中结果是"function"; 
//在Firefox3中结果是"object";

注:本人在safari中测试,其结果也是“function”。
3、IE6和IE7
在IE中对DOM元素使用typeof方法,得到的结果是“object”。如:
alert(typeof document.getElementsByTagName("body")[0].getAttribute); 
//结果是"object"

4、Safari 3
safari认为DOM元素的NodeList是一个函数,如:
alert(typeof document.body.childNodes); 
//结果是"function"

很明显,如果你要测试一个对象是否为函数,使用typeof方法并不能从真正意义上保证测试结果。那么,我们就需要一种在所有浏览器中都能保证测试结果的解决方案。我们知道function本身有apply()和call()两种方法,但这两个方法在IE中存在问题的函数中并不存在,试试下面的测试:
alert(typeof document.getElementsByTagName("body")[0].getAttribute.call) 
//在IE中结果是"undefined"

显然,我们不能利用这两个方法。

二、完美解决方案及实现过程
John Resig为我们提供了一个完美的解决方案,这个复杂但很稳定的判断一个对象是否为函数的方法如下:

function isFunction( fn ) { 
return !!fn && !fn.nodeName && fn.constructor != String && 
fn.constructor != RegExp && fn.constructor != Array && 
/function/i.test( fn + "" ); 
}

这个函数首先保证测试的对象存在,并将其序列化成含有“function”的字符串,这个是我们检测的基础(fn.constructor != String,fn.constructor != Array, and fn.constructor != RegExp)。另外,我们需要保证声明的函数不是一个DOM节点(fn.nodeName)。然后,我们就可以作toString测试。如果我们将一个函数转换成字符串,在一个浏览器中(fn+"")给我们的结果就像这样“function name(){...}”。现在,判断它是否为函数就很简单,仅仅只需要判断字符串中是否包含单词“function”。这很神奇,对于任何有问题的函数,在所有浏览器中都能得到我们所需要的结果。这个函数较之于传统的方法,运行速度有些不尽人意,作者建议我们保守使用。

John Resig 是jQuery库的开发者,相信使用该库的朋友们对该库简洁的语法和优秀的性能并不陌生。作者除追求代码简洁和性能高效之外,其尽善尽美的精神也让人叹服。如果你是一个完美主义者,相信此文对你很有帮助。

Javascript 相关文章推荐
JS实现向表格行添加新单元格的方法
Mar 30 Javascript
jQuery实现浮动层随浏览器滚动条滚动的方法
Sep 22 Javascript
BOM系列第二篇之定时器requestAnimationFrame
Aug 17 Javascript
JS实现HTML标签转义及反转义
Apr 14 Javascript
Javascript for in的缺陷总结
Feb 03 Javascript
JS实现移动端实时监听输入框变化的实例代码
Apr 12 Javascript
详解webpack打包vue时提取css
May 26 Javascript
Vue 菜单栏点击切换单个class(高亮)的方法
Aug 22 Javascript
实现Vue的markdown文档可以在线运行的方法示例
Dec 11 Javascript
详解一个基于套接字实现长连接的express
Mar 28 Javascript
React生命周期原理与用法踩坑笔记
Apr 28 Javascript
vue实现树状表格效果
Dec 29 Vue.js
javascript 控制 html元素 显示/隐藏实现代码
Sep 01 #Javascript
jsTree树控件(基于jQuery, 超强悍)[推荐]
Sep 01 #Javascript
JavaScript 继承详解 第一篇
Aug 30 #Javascript
JavaScript 应用技巧集合[推荐]
Aug 30 #Javascript
用cssText批量修改样式
Aug 29 #Javascript
javascript appendChild,innerHTML,join性能比较代码
Aug 29 #Javascript
IE 条件注释详解总结(附实例代码)
Aug 29 #Javascript
You might like
PHP Session机制简介及用法
2014/08/19 PHP
基于ThinkPHP+uploadify+upload+PHPExcel 无刷新导入数据
2015/09/23 PHP
Zend Framework框架教程之Zend_Db_Table_Rowset用法实例分析
2016/03/21 PHP
浅谈PHP封装CURL
2019/03/06 PHP
jquery 表单进行客户端验证demo
2009/08/24 Javascript
理解Javascript_03_javascript全局观
2010/10/11 Javascript
javascript定义函数的方法
2010/12/06 Javascript
jQuery 瀑布流 浮动布局(一)(延迟AJAX加载图片)
2012/05/23 Javascript
jQuery如何实现点击页面获得当前点击元素的id或其他信息
2014/01/09 Javascript
简单学习JavaScript中的for语句循环结构
2015/11/10 Javascript
浅谈jquery采用attr修改form表单enctype不起作用的问题
2016/11/25 Javascript
bootstrap导航条实现代码
2016/12/28 Javascript
解决vue-cli中stylus无法使用的问题方法
2017/06/19 Javascript
微信上传视频文件提示(推荐)
2018/11/22 Javascript
详解jenkins自动化部署vue
2019/05/14 Javascript
详解vue-cli3开发Chrome插件实践
2019/05/29 Javascript
详解mpvue实现对苹果X安全区域的适配
2019/07/31 Javascript
VUE使用axios调用后台API接口的方法
2020/08/03 Javascript
vue的hash值原理也是table切换实例代码
2020/12/14 Vue.js
Django ORM框架的定时任务如何使用详解
2017/10/19 Python
使用实现pandas读取csv文件指定的前几行
2018/04/20 Python
PyQt5实现拖放功能
2018/04/25 Python
Python之csv文件从MySQL数据库导入导出的方法
2018/06/21 Python
python中@property和property函数常见使用方法示例
2019/10/21 Python
Python 可视化神器Plotly详解
2020/12/26 Python
python 基于selectors库实现文件上传与下载
2020/12/31 Python
美国职棒大联盟官方网上商店:MLBShop.com
2017/11/12 全球购物
中国跨境在线时尚零售商:Bellelily
2018/04/06 全球购物
村党支部换届选举方案
2014/05/02 职场文书
党的群众路线教育实践方案
2014/05/11 职场文书
农村党支部书记党群众路线四风问题整改措施
2014/09/26 职场文书
2015年会计个人工作总结
2015/04/02 职场文书
2019中秋节祝福语大全,提前收藏啦
2019/09/10 职场文书
为什么代码规范要求SQL语句不要过多的join
2021/06/23 MySQL
JavaScript与JQuery框架基础入门教程
2021/07/15 Javascript
【海涛dota】偶遇拉娜娅 质量局德鲁伊第一视角解说
2022/04/01 DOTA