在JavaScript中typeof的用途介绍

JavaScript 中的 typeof 其实非常复杂,它可以用来做很多事情,但同时也有很多怪异的表现。

本文列举出了它的多个用法,而且还指出了存在的问题以及解决办法。

阅读本文的前提是,你现在应该已经知道原始值和对象值的区别了。

检查一个变量是否存在,是否有值
typeof在两种情况下会返回 "undefined":

1.变量没有被声明

2.变量的值是 undefined

例如:


代码如下:

> typeof undeclaredVariable === "undefined"
true

> var declaredVariable;
> typeof declaredVariable
'undefined'

> typeof undefined
'undefined'

还有其他办法检测某个值是否是 undefined:

代码如下:

> var value = undefined;
> value === undefined
true

但这种方法如果使用在一个未声明的变量上的时候,就会抛出异常,因为只有 typeof 才可以正常检测未声明的变量的同时还不报错:

代码如下:

> undeclaredVariable === undefined
ReferenceError: undeclaredVariable is not defined

注意:未初始化的变量,没有被传入参数的形参,不存在的属性,都不会出现上面的问题,因为它们总是可访问的,值总是 undefined:

代码如下:

> var declaredVariable;
> declaredVariable === undefined
true

> (function (x) { return x === undefined }())
true

> ({}).foo === undefined
true

译者注:因此,如果想检测一个可能没有被声明的全局变量是否存在,也可以使用if(window.maybeUndeclaredVariable){}。

问题:typeof 在完成这样的任务时显得很繁杂.

解决办法:这样的操作不是很常见,所以有人觉的没必要再找更好的解决办法了。 不过也许有人会提出一个专门的操作符:

代码如下:

> defined undeclaredVariable
false

> var declaredVariable;
> defined declaredVariable
false

或者,也许有人还需要一个检测变量是否被声明的操作符:

代码如下:

> declared undeclaredVariable
false

> var declaredVariable;
> declared declaredVariable
true

译者注:在 perl 里,上面的 defined 操作符相当于 defined(),上面的 declared 操作符相当于 exists()。

判断一个值不等于 undefined 也不等于 null
问题:如果你想检测一个值是否被定义过(值不是 undefined 也不是 null),那么你就遇到了 typeof 最有名的一个怪异表现(被认为是一个 bug):typeof null 返回了 "object":

代码如下:

> typeof null
'object'

译者注:这只能说是最初的 JavaScript 实现的 bug,而现在标准就是这样规范的。V8 曾经修正并实现过 typeof null === "null",但最终证明不可行。http://wiki.ecmascript.org/doku.php?id=harmony:typeof_null。

(译注:typeof 在操作 null 时会返回 "object",这是 JavaScript 语言本身的 bug。不幸的是,这个 bug 永远不可能被修复了,因为太多已有的代码已经依赖了这样的表现。但是 null 到底是不是 对象呢?stackoverflow 有关于这个问题的讨论:http://stackoverflow.com/questions/801032/null-object-in-javascript/7968470#7968470@justjavac)

解决办法:不要使用 typeof 来做这项任务,用下面这样的函数来代替:

代码如下:

function isDefined(x) {
    return x !== null && x !== undefined;
}

另一个可能性是引入一个 “默认值运算符”,在 myValue 未定义的情况下,下面的表达式会返回 defaultValue:

代码如下:

myValue ?? defaultValue

上面的表达式等价于:

代码如下:

(myValue !== undefined && myValue !== null) ? myValue : defaultValue

又或者:

代码如下:

myValue ??= defaultValue

其实是下面这条语句的简化:

代码如下:

myValue = myValue ?? defaultValue

当你访问一个嵌套的属性时,比如 bar,你或许会需要这个运算符的帮助:

代码如下:

obj.foo.bar

如果 obj 或者 obj.foo 是未定义的,上面的表达式会抛出异常。 一个运算符 .?? 可以让上面的表达式在遍历一层一层的属性时,返回第一个遇到的值为 undefined 或 null 的属性:

代码如下:

obj.??foo.??bar

上面的表达式等价于:

代码如下:

(obj === undefined || obj === null) ? obj
    : (obj.foo === undefined || obj.foo === null) ? obj.foo
        : obj.foo.bar

区分对象值和原始值

下面的函数用来检测 x 是否是一个对象值:

代码如下:

function isObject(x) {
    return (typeof x === "function"
            || (typeof x === "object" && x !== null));
}

问题:上面的检测比较复杂,是因为 typeof 把函数和对象看成是不同的类型,而且 typeof null 返回 "object".

解决办法:下面的方法也经常用于检测对象值:

代码如下:

function isObject2(x) {
    return x === Object(x);
}

警告:你也许认为这里可以使用 instanceof Object 来检测,但是 instanceof 是通过使用使用一个对象的原型来判断实例关系的,那么没有原型的对象怎么办呢:

代码如下:

> var obj = Object.create(null);
> Object.getPrototypeOf(obj)
null

obj 确实是一个对象,但它不是任何值的实例:

代码如下:

> typeof obj
'object'
> obj instanceof Object
false

在实际中,你可能很少遇到这样的对象,但它的确存在,而且有它的用途。

译者注:Object.prototype 就是唯一的一个内置的,没有原型的对象。

代码如下:

>Object.getPrototypeOf(Object.prototype)
null
>typeof Object.prototype
'object'
>Object.prototype instanceof Object
false

原始值的类型是什么?
typeof 是最好的用来查看某个原始值的类型的方式。

代码如下:

> typeof "abc"
'string'
> typeof undefined
'undefined'

问题:你必须知道 typeof null 的怪异表现。

代码如下:

> typeof null  // 要小心!
'object'

解决办法:下面的函数可以修复这个问题(只针对这个用例)。

代码如下:

function getPrimitiveTypeName(x) {
    var typeName = typeof x;
    switch(typeName) {
        case "undefined":
        case "boolean":
        case "number":
        case "string":
            return typeName;
        case "object":
            if (x === null) {
                return "null";
            }
        default: // 前面的判断都没通过
            throw new TypeError("参数不是一个原始值: "+x);
    }
}

更好的解决办法:实现一个函数 getTypeName(),除了可以返回原始值的的类型,还可以返回对象值的内部 [[Class]] 属性。 这里讲了如何实现这个函数(译者注:jQuery 中的 $.type 就是这样的实现)

某个值是否是函数
typeof 可以用来检测一个值是否是函数。

代码如下:

> typeof function () {}
'function'
> typeof Object.prototype.toString
'function'

原则上说,instanceof Function 也可以进行这种需求的检测。 乍一看,貌似写法还更加优雅。 但是,浏览器有一个怪癖:每一个框架和窗口都有它自己的全局变量。 因此,如果你将某个框架中的对象传到另一个框架中,instanceof 就不能正常工作了,因为这两个框架有着不同的构造函数。 这就是为什么 ECMAScript5 中会有Array.isArray() 方法的原因。 如果有一个能够跨框架的,用于检查一个对象是否是给定的构造函数的实例的方法的话,那会很好。 上述的 getTypeName() 是一个可用的变通方法,但也许还有一个更根本的解决方案。

综述
下面提到的,应该是目前 JavaScript 中最迫切需要的,可以代替一些 typeof 目前职责的功能特性:

•isDefined() (比如 Object.isDefined()): 可以作为一个函数或者一个运算符

•isObject()

•getTypeName()

•能够跨框架的,检测一个对象是否是指定的构造函数的实例的机制

检查某个变量是否已经被声明这样的需求,可能没那么必要有自己的运算符。

(0)

相关推荐

  • 学习 C++能带给我们什么

    C++准确说是一门中级语言,介于汇编和高级语言之间吧,要求程序员了解计算机的内部数据存储.个人认为,作为学生还是花功夫学C++,因为<设计模式><数据结构>这些课程基本上还是C++应付的比较好(我的切身体会),学习 C++,认真阅读c++ primer,而后配合 The ADAPTIVE Communication Environment (ACE)了解设计模式, 再看看<深入浅出STL>,就会发现其他语言都一样,不变的是思想本身. 在以下领域,C++有着根本性的优势:

  • 深入解析C语言中typedef的四个用途

    用途一: 定义一种类型的别名,而不只是简单的宏替换.可以用作同时声明指针型的多个对象.比如:char* pa, pb; // 这多数不符合我们的意图,它只声明了一个指向字符变量的指针, 和一个字符变量:以下则可行:typedef char* PCHAR; // 一般用大写PCHAR pa, pb; // 可行,同时声明了两个指向字符变量的指针虽然:char *pa, *pb; 也可行,但相对来说没有用typedef的形式直观,尤其在需要大量指针的地方,typedef的方式更省事. 用途二:用在旧

  • 在JavaScript中typeof的用途介绍

    JavaScript 中的 typeof 其实非常复杂,它可以用来做很多事情,但同时也有很多怪异的表现. 本文列举出了它的多个用法,而且还指出了存在的问题以及解决办法. 阅读本文的前提是,你现在应该已经知道原始值和对象值的区别了. 检查一个变量是否存在,是否有值typeof在两种情况下会返回 "undefined": 1.变量没有被声明 2.变量的值是 undefined 例如: 复制代码 代码如下: > typeof undeclaredVariable === "u

  • 详解JavaScript中Arguments对象用途

    目录 前言 Arguments 的基本概念 Arguments 的作用 获取实参和形参的个数 修改实参值 改变实参的个数 检测参数合法性 函数的参数个数不确定时,用于访问调用函数的实参值 遍历或访问实参的值 总结 在实际开发中,Arguments 对象非常有用.灵活使用 Arguments 对象,可以提升使用函数的灵活性,增强函数在抽象编程中的适应能力和纠错能力. JavaScript 中 Arguments 对象的用途总结. 前言 相信我们很多人在代码开发的过程中都使用到过一个特殊的对象 --

  • 详解JavaScript中typeof与instanceof用法

    今天写JS代码,遇到动态生成多个名称相同的input复选按钮 需要判断其是否是数组,用到了if (typeof(document.MapCheckMgr.checkid)!="undefined") 以前用得少,就顺便查了一下关于typeof的那些事 typeof用以获取一个变量或者表达式的类型,typeof一般只能返回如下几个结果: number,boolean,string,function(函数),object(NULL,数组,对象),undefined. 如: alert(ty

  • javascript中typeof的使用示例

    复制代码 代码如下: <html> <head> <title>javascript中typeof的使用</title> <script> //1.基本类型 var x = 123; var y = "abc"; var z = true; //alert(typeof x);//number //alert(typeof y);//string //alert(typeof z);//boolean //2.引用类型,类型是

  • Javascript中typeof 用法小结

    在js里用到数组,比如 多个名字相同(的)input, 若是动态生成(的), 提交时就需要判断其是否是数组. if(document.mylist.length != "undefined" ) {} 这个用法有误. 正确(的)是 if( typeof(document.mylist.length) != "undefined" ) {} 或 if( !isNaN(document.mylist.length) ) {} typeof(的)运算数未定义,返回(的)就

  • JavaScript中的ArrayBuffer详细介绍

    相信每一个 javascript 学习者,都会去了解 JS 的各种基本数据类型,数组就是数据的组合,这是一个很基本也十分简单的概念,他的内容没多少,学好它也不是件难事情.但是本文着重要介绍的并不是我们往常看到的 Array,而是 ArrayBuffer. 我写的很多东西都是因为要完成某些特定的功能而刻意总结的,可以算是备忘,本文也是如此!前段时间一直在研究 Web Audio API 以及语音通信相关的知识,内容侧重于音频流在 AudioContext 各个节点之间的流动情况,而现在要摸清楚音频

  • JavaScript中的异常捕捉介绍

    与Java语言相同,JavaScript可以通过throw语句来抛出异常.与Java语言不同的是,JavaScript中可以通过throw语句抛出所有类型的值,而不仅止于抛出错误对象. 复制代码 代码如下: //Throw an Error object. try{   throw new Error("Message in Error Object"); }catch(e){     console.log(e);//Error: Message in Error Object }

  • JavaScript中内存泄漏的介绍与教程(推荐)

    本文主要给大家详细介绍了关于JavaScript中内存泄漏的相关内容,文中介绍的非常详细,对大家具有一定的参考学习价值,下面来一起看看详细的介绍: 一.什么是内存泄漏? 程序的运行需要内存.只要程序提出要求,操作系统或者运行时(runtime)就必须供给内存. 对于持续运行的服务进程(daemon),必须及时释放不再用到的内存.否则,内存占用越来越高,轻则影响系统性能,重则导致进程崩溃. 不再用到的内存,没有及时释放,就叫做内存泄漏(memory leak). 有些语言(比如 C 语言)必须手动

  • Javascript中的call()方法介绍

    在Mozilla的官网中对于call()的介绍是: 复制代码 代码如下: call() 方法在使用一个指定的this值和若干个指定的参数值的前提下调用某个函数或方法. Call() 语法 复制代码 代码如下: fun.call(thisArg[, arg1[, arg2[, ...]]]) Call() 参数 thisArg 复制代码 代码如下: 在fun函数运行时指定的this值.需要注意的是,指定的this值并不一定是该函数执行时真正的this值,如果这个函数处于非严格模式下,则指定为nul

  • javascript中onclick(this)用法介绍

    this指触发事件的对象 复制代码 代码如下: <input id="myinput" type="text" value="javascript中onclick中的this" onclick="javascript:test(this);"/> 复制代码 代码如下: function test(obj){ alert(obj); //[object HTMLInputElement] alert(obj.id);

随机推荐