从零开始学JavaScript(二)- 理解函数(续)

 一些随手写下的笔记,方便以后参考用,可能会有一些错误。

2016-01-24
关于DOM选择器querySelector与querySelectorAll的用法;
document.querySelector选择器,选取的元素是默认的第一元素;如果没有匹配项,返回null。
document.querySelectorAll选取的元素是该元素所有数组(类似Array);如果没有匹配项,返回空的nodelist(节点数组[])。
2016-01-26
匿名函数与构造函数(javascript高级程序设计110页详细解释)
函数即是对象。每个函数都是function类型的实例,而且都是与其他引用类型一样,具有属性与方法。由于函数是对象,因此函数名实际上也是一个指向函数对象的指针,不会与某个函数绑定。(对象的概念)
函数表达式;
funtion Name (参数A) {
代码片段function_body;
}
Name(参数B);
注:2016-01-27修改[有一说法是 funtion Name (参数A){function_body;} 使用function关键字声明一个函数,再指定一个函数名,叫函数声明。而我的笔记中写的是 自定义函数,待考究!] 
参数有实际参数与形式参数之分,实际参数是调用函数时,给予函数的真实传值(参数B),形式参数是写在函数表达式中的自定义参数(参数A)。
函数直接量;
var myName = function (参数A){
代码片段function_body;
};/*注意分号,函数结尾处分号的差别,因为声明变量时用分号区别*/
函数直接量与函数表达式写法相似,但不等同与(待考究);
注:2016-01-27修改[同上,理解有误:使用function关键字声明一个函数,但未给函数命名,最后将匿名函数赋予一个变量,叫函数表达式,这是最常见的函数表达式语法形式。]
例:
myName(参数B);
var myName = function (参数A){
代码片段function_body;
};/*执行错误,变量myName还未保存对函数的引用,函数调用必须在函数表达式之后*/
与:
Name(参数B);
funtion Name (参数A) {
代码片段function_body;
};/*执行正确,并且调用函数时,优先读取函数表达式;*/
解释:函数声明和函数表达式不同之处在于:一、Javascript引擎在解析javascript代码时会‘函数声明提升’(Function declaration Hoisting)当前执行环境(作用域)上的函数声明,而函数表达式必须等到Javascirtp引擎执行到它所在行时,才会从上而下一行一行地解析函数表达式。二、函数表达式后面可以加括号立即调用该函数,函数声明不可以,只能以myName()形式调用 。
匿名函数;注:2016-01-27新增内容
function () {}; 使用function关键字声明一个函数,但未给函数命名,所以叫匿名函数,匿名函数属于函数表达式,匿名函数有很多作用,赋予一个变量则创建函数,赋予一个事件则成为事件处理程序或创建闭包等。
function构造函数;
var anName = new function ("参数A","参数A",“return function_body”)
高级程序设计说明:从技术的角度讲,这是一个函数表达式。但是不推荐读者使用这种方法定义函数,因为这种语法会导致解析两次代码,第一次解析常规的ECMAScript代码,第二次解析传入构造函数中的字符串。从而影响性能,不过这种语法对于理解“函数是对象,函数名是指针”的概念倒是非常直观的。
对于构造函数的示例:
(部分摘自开源中国)
在JavaScript运行时中有一个内建的类叫做Function,用function关键字声明一个函数其实是创建Function类对象的一种简写形式,所有的函数都拥有Function类所有的方法,例如call、apply、bind等等,可以通过instanceof关键字来验证这个说法。 
既然Function是一个类,那么它的构造函数就是Function(它本身也是Function类的对象),应该可以通过new关键字来生成一个函数对象。
示例一、 
new Function ([arg1[, arg2[, ... argN]],] function_body) 
其中arg1, arg2, ... argN是字符串,代表参数名称,function_body也是字符串,表示函数体,前面的参数名称是可多可少的,Function的构造函数会把最后一个参数当做函数体,前面的都当做参数处理。 
示例二、  
var func1 = new Function('name', 'return "Hello, " + name + "!";'); 
func1('Ghostheaven'); // Hello, Ghostheaven! 
以上方法就通过Function构造了一个函数,这个函数跟其他用function关键字声明的函数一模一样。 
为什么会存在这样一个Function类,“存在的即是合理的”,Function类有它独特的用途,你可以利用它动态地生成各种函数逻辑,或者代替eval函数的功能,而且能保持当前环境不会被污染。 
重点:由于函数名仅仅是指向函数的指针,因此函数名与包含函数对象指针的其他变量没有不同,一个函数可以有多个函数名。
例:
function heName ( add,bob){
return add+bob ;
}
heName(10,5);
var herName = heName ;
alert(herName(10,10));
sum = null;
alert(herName(10,20));
(示例摘自开源中国)
fnName();
function fnName(){
    ...
}//正常,因为‘提升’了函数声明,函数调用可在函数声明之前  
fnName();
var fnName=function(){
    ...
}//报错,变量fnName还未保存对函数的引用,函数调用必须在函数表达式之后
var fnName=function(){
    alert('Hello World');
}();//函数表达式后面加括号,当javascript引擎解析到此处时能立即调用函数
function fnName(){
    alert('Hello World');
}();//不会报错,但是javascript引擎只解析函数声明,忽略后面的括号,函数声明不会被调用
function(){
    console.log('Hello World');    
}();//语法错误,虽然匿名函数属于函数表达式,但是未进行赋值操作,
//所以javascript引擎将开头的function关键字当做函数声明,报错:要求需要一个函数名 。
区别( function(){…} )()和( function (){…} () )这两种立即执行函数的写法,要在函数体后面加括号就能立即调用,则这个函数必须是函数表达式,不能是函数声明。
例:
(function(a){
    console.log(a);   //firebug输出123,使用()运算符
})(123);
(function(a){
    console.log(a);   //firebug输出1234,使用()运算符
}(1234));
!function(a){
    console.log(a);   //firebug输出12345,使用!运算符
}(12345);
+function(a){
    console.log(a);   //firebug输出123456,使用+运算符
}(123456);
-function(a){
    console.log(a);   //firebug输出1234567,使用-运算符
}(1234567);
var fn=function(a){
    console.log(a);   //firebug输出12345678,使用=运算符
}(12345678) 
可以看到输出结果,在function前面加!、+、 -甚至是逗号等到都可以起到函数定义后立即执行的效果,而()、!、+、-、=等运算符,都将函数声明转换成函数表达式,消除了javascript引擎识别函数表达式和函数声明的歧义,告诉javascript引擎这是一个函数表达式,不是函数声明,可以在后面加括号,并立即执行函数的代码。
加括号是最安全的做法,因为!、+、-等运算符还会和函数的返回值进行运算,有时造成不必要的麻烦
不过这样的写法有什么用呢?
javascript中没用私有作用域的概念,如果在多人开发的项目上,你在全局或局部作用域中声明了一些变量,可能会被其他人不小心用同名的变量给覆盖掉,根据javascript函数作用域链的特性,可以使用这种技术可以模仿一个私有作用域,用匿名函数作为一个“容器”,“容器”内部可以访问外部的变量,而外部环境不能访问“容器”内部的变量,所以( function(){…} )()内部定义的变量不会和外部的变量发生冲突,俗称“匿名包裹器”或“命名空间”。
JQuery使用的就是这种方法,将JQuery代码包裹在( function (window,undefined){…jquery代码…} (window)中,在全局作用域中调用JQuery代码时,可以达到保护JQuery内部变量的作用。
原文地址:https://www.cnblogs.com/zero-Alex/p/5163238.html