主要内容

  1. 分析函数的四种调用形式
  2. 弄清楚函数中this的意义
  3. 明确构造函对象的过程
  4. 学会使用上下文调用函数

  了解函数的调用过程有助于深入学习与分析JavaScript代码. 本文是JavaScript高级这 个系列中的第三篇文章,主要介绍JavaScript中函数的四种使用形式.
  在JavaScript中,函数是一等公民,函数在JavaScript中是一个数据类型,而非像C#或
其他描述性语言那样仅仅作为一个模块来使用. 函数有四种调用模式,分别是:函数调用形
式、方法调用形式、构造器形式、以及apply形式. 这里所有的调用模式中,最主要的区别在
于关键字 this 的意义. 下面分别介绍这个几种调用形式.

一、函数调用形式

函数调用形式是最常见的形式,也是最好理解的形式. 所谓函数形式就是一般声明函数 后直接调用即是. 例如:

 // 声明一个函数,并调用
function func() {
alert("Hello World");
}
func();

或者

 // 使用函数的Lambda表达式定义函数,然后调用
var func = function() {
alert("你好,传智播客");
};
func();

这两段代码都会在浏览器中弹出一个对话框,显示字符串中的文字. 这个就是函数调用.

  可以发现函数调用很简单,就是平时学习的一样. 这里的关键是,在函数调用模式中, 函数里的 this 关键字指全局对象,如果在浏览器中就是 window 对象. 例如:

 var func = function() {
alert(this);
};
func();

此时,会弹出对话框,打印出 [object Window]

二、方法调用模式

  函数调用模式很简单,是最基本的调用方式. 但是同样的是函数,将其赋值给一个对 象的成员以后,就不一样了. 将函数赋值给对象的成员后,那么这个就不在称为函数,而
应该叫做方法. 例如:

 // 定义一个函数
var func = function() {
alert("我是一个函数么?");
};
// 将其赋值给一个对象
var o = {};
o.fn = func; // 注意这里不要加圆括号
// 调用
o.fn();

此时,o.fn 则是方法,不是函数了. 实际上 fn 的方法体与 func 是一模一样的,但是这
里有个微妙的不同. 看下面的代码:

 // 接上面的代码
alert(o.fn === func);

打印结果是 true ,这个表明两个函数是一样的东西. 但是修改一下函数的代码:

 // 修改函数体
var func = function() {
alert(this);
};
var o = {};
o.fn = func;
// 比较
alert(o.fn === func);
// 调用
func();
o.fn();

  这里的运行结果是,两个函数是相同的,因此打印结果是 true. 但是由于两个函数的调用
是不一样的,func 的调用,打印的是 [object Window],而 o.fn 的打印结果是 [object Object].

  这里便是函数调用与方法调用的区别. 函数调用中,this 专指全局对象 window,而
在方法中 this 专指当前对象. 即 o.fn 中的 this 指的就是对象 o.

三、构造器调用模式

  同样是函数,在单纯的函数模式下,this 表示 window;在对象方法模式下,this 指 的是当前对象. 除了这两种情况,JavaScript 中函数还可以是构造器. 将函数作为构造器
来使用的语法就是在函数调用前面加上一个 new 关键字. 如代码:

 // 定义一个构造函数
var Person = function() {
this.name = "传智播客";
this.sayHello = function() {
alert("你好,这里是" + this.name);
};
};
// 调用构造器,创建对象
var p = new Person();
// 使用对象
p.sayHello();

  上面的案例首先创建一个构造函数Person,然后使用构造函数创建对象p. 这里使用 new 语法. 然后在使用对象调用sayHello()方法. 这个使用构造函数创建对象的案例比较简单.
从案例可以看到,此时 this 指的是对象本身.
  除了上面简单的使用以外,函数作为构造器还有几个变化. 分别为:

   1、 所有需要由对象使用的属性,必须使用 this 引导;
   2、 函数的 return 语句意义被改写,如果返回非对象,就返回this;

3.1 构造器中的 this

  我们需要分析创建对象的过程,方能知道 this 的意义. 如下面代码:

 var Person = function() {
this.name = "传智播客";
};
var p = new Person();

这里首先定义了函数 Person,下面分析一下整个执行:
  1、 程序在执行到这一句的时候,不会执行函数体,因此 JavaScript 的解释器并不知道这个
    函数的内容.

  2、 接下来执行 new 关键字,创建对象,解释器开辟内存,得到对象的引用,将新对象的引
    用交给函数.
  3、 紧接着执行函数,将传过来的对象引用交给 this. 也就是说,在构造方法中,this 就是
    刚刚被 new 创建出来的对象.

  4、 然后为 this 添加成员,也就是为对象添加成员.
  5、 最后函数结束,返回 this,将 this 交给左边的变量.

  分析过构造函数的执行以后,可以得到,构造函数中的 this 就是当前对象.

3.2 构造器中的 return

  在构造函数中 return 的意义发生了变化,首先如果在构造函数中,如果返回的是一个对 象,那么就保留原意. 如果返回的是非对象,比如数字、布尔和字符串,那么就返回 this,如
果没有 return 语句,那么也返回 this. 看下面代码:

 // 返回一个对象的 return
var ctr = function() {
this.name = "赵晓虎";
return {
name:"牛亮亮"
};
};
// 创建对象
var p = new ctr();
// 访问name属性
alert(p.name);

执行代码,这里打印的结果是"牛亮亮". 因为构造方法中返回的是一个对象,那么保留 return
的意义,返回内容为 return 后面的对象. 再看下面代码:

 // 定义返回非对象数据的构造器
var ctr = function() {
this.name = "赵晓虎";
return "牛亮亮";
};
// 创建对象
var p = new ctr();
// 使用
alert(p);
alert(p.name);

代码运行结果是,先弹窗打印[object Object],然后打印"赵晓虎". 因为这里 return 的是一个
字符串,属于基本类型,那么这里的 return 语句无效,返回的是 this 对象. 因此第一个打印的
是[object Object]而第二个不会打印 undefined.

四、apply调用模式

  除了上述三种调用模式以外,函数作为对象还有 apply 方法与 call 方法可以使用,这便是 第四种调用模式,我称其为 apply 模式.
  首先介绍 apply 模式,首先这里 apply 模式既可以像函数一样使用,也可以像方法一样使用
可以说是一个灵活的使用方法. 首先看看语法:

  函数名.apply(对象, 参数数组);

这里看语法比较晦涩,还是使用案例来说明:

  1、 新建两个 js 文件,分别为"js1.js"与"js2.js";
  2、 添加代码

 // js1.js 文件中
var func1 = function() {
this.name = "传智播客";
};
func1.apply(null);
alert(name); // js2.js 文件
var func2 = function() {
this.name = "传值播客";
};
var o = {};
func2.apply(o);
alert(o.name);

  3、 分别运行着两段代码,可以发现第一个文件中的 name 属性已经加载到全局对象 window 中;
    而第二个文件中的 name 属性是在传入的对象 o 中. 即第一个相当于函数调用,第二个相当
    于方法调用.

   这里的参数是方法本身所带的参数,但是需要用数组的形式存储在. 比如代码:

 // 一个数组的例子
var arr1 = [1,2,3,[4,5],[6,7,8]];
// 将其展开
var arr2 = arr1.conact.apply([], arr1);

  然后介绍一下 call 模式. call 模式与 apply 模式最大的不同在于 call 中的参数不用数组. 看下面代码就清楚了:

 // 定义方法
var func = function(name, age, sex) {
this.name = name;
this.age = age;
this.sex = sex;
};
// 创建对象
var o = {};
// 给对象添加成员
// apply 模式
var p1 = func.apply(o, ["赵晓虎", 19, "男"]);
// call 模式
var p2 = func.call(o, "赵晓虎", 19, "男");

上面的代码,apply 模式与 call 模式的结果是一样的.

  实际上,使用 apply 模式和 call 模式,可以任意的操作控制 this 的意义,在函数 js 的设 计模式中使用广泛. 简单小结一下,js 中的函数调用有四种模式,分别是:函数式、方法式、构造
器式和 apply 式. 而这些模式中,this 的含义分别为:在函数中 this 是全局对象 window,在方
法中 this 指当前对象,在构造函数中 this 是被创建的对象,在 apply 模式中 this 可以随意的
指定. 在 apply 模式中如果使用 null,就是函数模式,如果使用对象,就是方法模式.

五、案例   

  下面通过一个案例结束本篇吧.
案例说明:有一个div,id为dv,鼠标移到上面去高度增大2倍,鼠标离开恢复. 下面直接上js代码

 var dv = document.getElementById("dv");
var height = parseInt(dv.style.height || dv.offsetHeight);
var intervalId;
dv.onmouseover = function() {
// 停止已经在执行的动画
clearInterval(intervalId);
// 得到目标高度
var toHeight = height * 2;
// 获得当前对象
var that = this;
// 开器计时器,缓慢变化
intervalId = setInterval(function() {
// 得到现在的高度
var height = parseInt(dv.style.height || dv.offsetHeight);
// 记录每次需要变化的步长
var h = Math.ceil(Math.abs(height - toHeight) / 10);
// 判断变化,如果步长为0就停止计时器
if( h > 0 ) {
// 这里为什么要用that呢?思考一下吧
that.style.height = (height + h) + "px";
} else {
clearInterval(intervalId);
}
}, 20);
};
dv.onmouseout = function() {
// 原理和之前一样
clearInterval(intervalId);
var toHeight = height;
var that = this;
intervalId = setInterval(function() {
var height = parseInt(dv.style.height || dv.offsetHeight);
var h = Math.ceil(Math.abs(height - toHeight) / 10);
if( h > 0 ) {
that.style.height = (height - h) + "px";
} else {
clearInterval(intervalId);
}
}, 20);
};

JavaScript高级之函数的四种调用形式的更多相关文章

  1. javascript this 代表的上下文,JavaScript 函数的四种调用形式

    JavaScript 是一种脚本语言,支持函数式编程.闭包.基于原型的继承等高级功能.其中JavaScript 中的 this 关键字,就是一个比较容易混乱的概念,在不同的场景下,this会化身不同的 ...

  2. JS高级. 06 缓存、分析解决递归斐波那契数列、jQuery缓存、沙箱、函数的四种调用方式、call和apply修改函数调用方法

    缓存 cache 作用就是将一些常用的数据存储起来 提升性能 cdn //-----------------分析解决递归斐波那契数列<script> //定义一个缓存数组,存储已经计算出来 ...

  3. JS面向对象函数的四种调用模式

    函数的四种调用模式 概念 在 js 中,无论是函数, 还是方法, 还是事件, 还是构造器,...这些东西的本质都是函数 函数, 方法, 事件, 构造器,...只是所处的位置不同 这四种模式分别是 函数 ...

  4. 函数的四种调用模式.上下文调用.call.apply

    闭包:函数就是一个闭包,一个封闭的作用域;         返回函数,要返回多个函数就用一个对象封装一下,         立即执行函数+return 回调函数   JS动态创建的DOM,不会被搜索引 ...

  5. Javascript中函数的四种调用方式

    一.Javascript中函数的几个基本知识点: 1.函数的名字只是一个指向函数的指针,所以即使在不同的执行环境,即不同对象调用这个函数,这个函数指向的仍然是同一个函数. 2.函数中有两个特殊的内部属 ...

  6. javascript中函数的四种调用模式详解

    介绍函数四种调用模式前,我们先来了解一下函数和方法的概念,其实函数和方法本质是一样,就是称呼不一样而已.函数:如果一个函数与任何对象关系,就称该函数为函数.方法:如果一个函数作为一个对象属性存在,我们 ...

  7. js高级-函数的四种调用模式

    1.对象方法调用模式  方法内部的this指向当前调用者的对象d 定义类 (构造函数) function Dog (dogName){ //创建一个空对象   让空对象==this this.name ...

  8. JS函数的四种调用模式

    函数在js中具有四种身份,分别为函数.方法.构造函数.apply或call调用 函数调用    函数调用模式中this指全局对象(window) var f1 = function() { alert ...

  9. javascript函数的四种调用模式及其this关键字的区别

    方法调用模式: 当一个函数被保存为对象的一个属性时,我们称它为一个方法.当一个方法被调用时,this被绑定到该对象. //方法调用模式 var myObject = { value: 0 , incr ...

随机推荐

  1. redis-BOOK

    https://www.gitbook.com/book/gnuhpc/redis-all-about/details

  2. map的例子

    11.4 编写单词计数程序,忽略大小写和标点.例如,“example.”,“example,"和”Example“应该递增相同的计算器. #include<iostream> # ...

  3. linux 修改文件时间

    1.ls -l *.sh 2.touch -d "10/13/2013" *.sh [我想把所以的.sh文件修改到三个月前(2013年10月13)的时间.]3.ls -l *.sh ...

  4. (转)JSON基础入门

    原文地址:http://kb.cnblogs.com/page/43982/ JSON 基础简单地说,JSON 可以将 JavaScript 对象中表示的一组数据转换为字符串,然后就可以在函数之间轻松 ...

  5. magento性能优化的教程(非常详细)

    Magento是一套专业开源的电子商务系统,Magento设计得非常灵活,具有模块化架构体系和丰富的功能但有朋友会发现此模块用到了会发现非常的缓慢了,那么下面我们来看关于magento性能优化的例子. ...

  6. 用变量a给出下面的定义

    a)一个整型数(An integer)b) 一个指向整型数的指针(A pointer to an integer)  c) 一个指向指针的的指针,它指向的指针是指向一个整型数(A pointer to ...

  7. ios ReactiveViewModel

    项目中使用 ReactiveCocoa 一般都会嵌入ReactiveViewModel 或者 ReactiveCocoaLayout 联合处理UI.网络.动画.布局.窗口切换等,组合使用时威力惊人. ...

  8. Oracle抓取表结构的语句

    oracle ---------------------------------------   SELECT case when t.COLUMN_ID=1 then t.table_name en ...

  9. linux修改主机名(hostname)转载

    Linux修改主机名的方法 用hostname命令可以临时修改机器名,但机器重新启动之后就会恢复原来的值. #hostname   //查看机器名#hostname -i  //查看本机器名对应的ip ...

  10. MySQL中的datetime与timestamp比较

    引用:http://database.51cto.com/art/200905/124240.htm TIMESTAMP列的显示格式与DATETIME列相同.换句话说,显示宽度固定在19字符,并且格式 ...