浅谈Javascript

Javascript 中的变量作用域问题

Js中的变量作用域问题:

1、没有块级作用域。Js中的变量作用域不是以{}为界的,不像C/C++/Java。

如:

1 if(true){
2     var name = "qqyumidi";
3 }
4             
5 alert(name);  // 结果:qqyumidi

Js会将在if中定义的变量添加到当前的执行环境中,尤其在使用for循环时需要注意与其他语言的差异。

1 for(var i=0; i<10; i++){
2     ;
3 }
4 
5 alert(i);   // 结果:10 

JavaScript的数据类型

1、Js数据类型

在Js中,有五种基本数据类型:Undefined、Null、Number、String和Boolean。还有一种复杂数据类型Object。Object本质上是由一组无序的名值对组成。Js中不支持自己所创建的任何类型机制,所有值最终都是以上六种类型之一。Undefined类型只有一个值undefined,Null类型只有一个值null,Boolean类型有两个值true和false。

2、判断变量的数据类型

可以通过typeof操作符在一定程度上检测变量数据类型。typeof是操作符范畴,而不是函数。typeof操作符返回的结果是一个变量数据类型相应的字符串表示形式,如:String类型则返回"string",Boolean类型则返回"boolean",Undefined类型则返回"undefined",Null类型则返回"null"。typeof操作符返回的结果与Js数据类型并不完全一致,一共返回有六种结果,且其与数据类型对应关系如下:

Undefined => "undefined"
Null          =>  "object" (js认为null是对对象的空引用)

Number    =>  "number"
Boolean    =>  "boolean"
String       =>  "string"
Object      =>   "object" (除去函数)
               =>    "function" (函数)

由此可以看出,完全由typeof操作符来判断变量的数据类型是不可以的。同时对于Js中的变量,还存在如下情况:

1 var a;
2 alert(typeof a);  // result: "undefined"
3 alert(typeof b);  // result: "undefined"

如上例所示,对一个已经定义的变量但未初始化(其实本例中的var a等同于var a=undefined),和一个尚未定义的变量同时typeof的结果都是"undefined",由此也可以看出如果需要判断变量数据类型是否是Undefined直接通过typeof操作符也是不严谨的。

对于未声明的变量,如果直接在程序中用到了,除了typeof操作为生命变量外,都会出现错误提示,因为变量尚未申明,是不能直接用的。

因此在判断变量是否为Undefined时,最好还是通过if(typeof varname == "undefined")形式,而不是直接通过if(varname == undefined)形式。

那么现在问题出来了,如何比较好的判断一个变量是未定义还是已经定义但尚未赋值呢?

JavaScript arguments对象

1、在JavaScript中,arguments对象是比较特别的一个对象,实际上是当前函数的一个内置属性。arguments非常类似Array,但实际上又不是一个Array实例。可以通过如下代码得以证实(当然,实际上,在函数funcArg中,调用arguments是不必要写成funcArg.arguments,直接写arguments即可)。

复制代码
1 Array.prototype.testArg = "test";
2 function funcArg() {
3     alert(funcArg.arguments.testArg);  
4     alert(funcArg.arguments[0]);
5 }
6 
7 alert(new Array().testArg); // result: "test"
8 funcArg(10);                // result: "undefined"  "10"
复制代码

2、arguments对象的长度是由实参个数而不是形参个数决定的。形参是函数内部重新开辟内存空间存储的变量,但是其与arguments对象内存空间并不重叠。对于arguments和值都存在的情况下,两者值是同步的,但是针对其中一个无值的情况下,对于此无值的情形值不会得以同步。如下代码可以得以验证。

复制代码
 1 function f(a, b, c){
 2     alert(arguments.length);   // result: "2"
 3     a = 100;
 4     alert(arguments[0]);       // result: "100"
 5     arguments[0] = "qqyumidi";
 6     alert(a);                  // result: "qqyumidi"
 7     alert(c);                  // result: "undefined"
 8     c = 2012;
 9     alert(arguments[2]);       // result: "undefined"
10 }
11 
12 f(1, 2);
复制代码

3、由JavaScript中函数的声明和调用特性,可以看出JavaScript中函数是不能重载的

根据其他语言中重载的依据:"函数返回值不同或形参个数不同",我们可以得出上述结论:

第一:Javascript函数的声明是没有返回值类型这一说法的;

第二:JavaScript中形参的个数严格意义上来讲只是为了方便在函数中的变量操作,实际上实参已经存储在arguments对象中了。

另外,从JavaScript函数本身深入理解为什么JavaScript中函数是不能重载的:在JavaScript中,函数其实也是对象,函数名是关于函数的引用,或者说函数名本身就是变量。对于如下所示的函数声明与函数表达式,其实含以上是一样的(在不考虑函数声明与函数表达式区别的前提下),非常有利于我们理解JavaScript中函数是不能重载的这一特性。

复制代码
 1 function f(a){
 2     return a + 10;
 3 }
 4 
 5 function f(a){
 6     return a - 10;
 7 }
 8 
 9 // 在不考虑函数声明与函数表达式区别的前提下,其等价于如下
10 
11 var f = function(a){
12     return a + 10;
13 }
14 
15 var f = function(a){
16     return a - 10;
17 }
复制代码

4、arguments对象中有一个非常有用的属性:callee。arguments.callee返回此arguments对象所在的当前函数引用。在使用函数递归调用时推荐使用arguments.callee代替函数名本身。

如下:

复制代码
1 function count(a){
2     if(a==1){
3         return 1;
4     } 
5     return a + arguments.callee(--a);
6 }
7 
8 var mm = count(10);
9 alert(mm);

文章部分素材取自 :  http://www.cnblogs.com/lwbqqyumidi/archive/2012/07/12/2587903.html

http://www.cnblogs.com/lwbqqyumidi/archive/2012/12/02/2798616.html


你可能感兴趣的:(初识Javascript)