引言:所谓编程就是将一组需求分解成一组函数与数据结构的技能。
函数也是对象
对象是什么?key\value
的集合并且有一个连到原型对象的隐藏链接。函数这个对象可以被运行,我们可以认为它具有一个被调用的属性,而()
被称为调用符,每一个函数都自带了this指针,指向调用者。
输入的参数和定义的参数不匹配会怎么样
如果实际参数过多,那么多余的将会被忽略,否则缺少的值将被替换为undefined
方法调用模式
当一个函数被保存为对象的属性时,被称为方法,方法被调用时,this指针就绑定了当前的调用者,再使用()
时也就是方法调用。代码:
var myObject = {
value:0,
increment: function (num){
this.value += typeof num === 'number' ? num : 1;
}
};
myObject.increment();//方法调用,this指针绑定到了myObject,同时没有传参数,相当于传入undefined
出问题的调用模式--函数调用
var sum = add(3,4);
这时候的this指针指向了window
使用这种方式时,内部函数的this跳过阻拦,不顾外部函数的感受,直接暴露在了全局环境中。
var myObject = {
value:1,
increment: function (num){
this.value += typeof num === 'number' ? num : 1;
}
};
myObject.double = function(){
var help = function(){
this.value = this.value *2;
};
help();
};
myObject.double();
上述代码为myObject对象添加了一个double方法,在这个方法内声明了一个内部函数help(),按照字面意思应该是让myObject对象的value值变化,可是实际运行后,并没有改变myObject.value的值,因为这样的调用把help暴露在全局环境中,那么this就指向了window而不是myObject,改进如下:
myObject.double = function(){
var that = this;//因为double是方法调用,所以this指向调用者myObject
var help = function(){
that.value = that.value *2;//help函数即使暴露在全局环境下,that指针依旧指向myObject
};
help();
};
myObject.double();
构造器调用模式
在这门语言中的new不要固执地和其他语言搞混了,如果在一个函数前面加上new
的话,那么将创建一个隐藏链接到该函数的prototype成员的新对象。(确实不好理解)
var Quo = function(){
this.status = "open";
};
//给所有的Quo实例提供get_status方法,想想为什么(添加到原型上)
Quo.prototype.get_status = function(){
console.log(this.status);
}
var myQuo = new Quo();
myQuo.get_status();
Apply调用模式
之前说过,JavaScript是一门函数式面向对象编程语言,函数也是对象,所以它拥有方法。
apply方法允许我们选择this值,接受两个参数,第一个是将与this绑定的值,第二个是一个参数数组,供原来的方法使用。
var name1 = {
key:1
};
var name2 = {
key:2
}
console.log(Math.abs.apply(name1,[name1.key]));
上述代码在两个命名空间下分别设置了key属性的值,我们可以看到,在调用Math.abs方法时,我们通过apply方法规定了其作用域在name1空间下,参数数组目前只包含一个值,为name1下的key,结果也显而易见,为1,再举一个例子:
var test = {
status:"test-open"
}
Quo.prototype.get_status.apply(test);
新创建了一个对象,拥有status属性,调用时,沿着Quo对象的原型链找到了get_status
方法,通过apply将它内部的this指针指向了test作用域
,所以运行结果和test.status
别无二致。
返回值
我们都知道遇到return语句,函数会立即终止,并且返回相应值,如果没有指定,则返回undefined,需要理解下面语句:
var demo = new Object();
赋值号右边的部分生成一个新对象,并且返回了this指针,那么demo便是这个新对象的引用。
给类型增加方法
Function.prototype.method = function(name,func){
this.prototype[name]=func;
return this;
};
给Function对象的原型增加一个method方法,那么对于Function的所有实例都拥有这个方法,调用的时候,会将func绑定到它原型的name属性上,先前已经说过,如果name属性还不存在,那么便会自动创建它,所以该method的作用等价于object.prototype.name = func
,以后书写的时候就可以省略prototype了。