JS编程建议——59:推荐动态调用函数

简介: 59:推荐动态调用函数

建议59:推荐动态调用函数
调用函数更便捷的方式是使用Function对象的call和apply方法。apply与call方法在本质上没有太大区别,只不过它们传递给函数的参数方式不同, apply是以数组形式进行参数传递,而call方法可以同时传递多个值。
如果某个函数仅能够接收多个参数列表,而现在希望把一个数组的所有元素作为参数进行传递,那么使用apply方法就显得非常便利。
function max(){

var m = Number.NEGATIVE_INFINITY; // 声明一个负无穷大的数值
for( var i = 0; i < arguments.length; i ++ ){
    if( arguments[i] > m )
    m = arguments[i];
}
return m;

}
var a = [23, 45, 2, 46, 62, 45, 56, 63];
var m = max.apply( Object, a );
alert( m ); //63
在上面的示例中,首先定义一个函数来计算所传递实参的最大值。由于该函数仅能够接收多个数值参数,所以通过apply方法动态调用max()函数,然后把它绑定为Object对象的一个方法,并借机把一个数组传递给它,最后返回此函数的运行值。如果没有apply方法,想使用max()函数来计算数组中最大元素值,就需要把数组的所有元素读取出来,然后再传递给函数,显然这种做法是费力不讨好的。
实际上,也可以把数组元素通过apply方法传递给系统对象Math的max方法来计算数组的最大元素值。
var a = [23, 45, 2, 46, 62, 45, 56, 63]; // 声明并初始化数组
var m = Math.max.apply( Object, a ); // 调用系统函数max
alert( m ); //63
使用call和apply方法可以把一个函数转换为方法传递给某个对象。这种行为只是临时的,函数最终并没有作为对象的方法而存在,当函数被调用后,该对象方法会自动被注销。下面的示例具体地说明了这种行为。
function f(){}
f.call( Object );
Object.f();
call和apply方法能够更改对象的内部指针,即改变对象的this指向的内容,这在面向对象的编程过程中是非常有用的。
var x = "o";
function a(){

this.x = "a";

}
function b(){

this.x = "b";

}
function c(){

alert( x );

}
function f(){

alert( this.x );

}
f(); //字符o,即全局变量x的值。this此时指向window对象
f.call( window ); f.call( new a() ); //字符a,即函数a内部的局部变量x的值。this此时指向函数a
f.call( new b() ); //字符b,即函数b内部的局部变量x的值。this此时指向函数b
f.call( c ); /undefined,即函数c内部的局部变量x的值,但是该函数并没有定义x变量,所以返回没有定义。this此时指向函数c/
通过上面示例的比较,能够很直观地发现,函数f内部的this关键字会随着所绑定的对象不同而指向不同的对象。因此,利用call或apply方法能够改变函数内部指针指向所绑定的对象,从而实现属性或方法继承。
function f(){

this.a ="a";
this.b = function(){
    alert("b");
}

}
function e(){

f.call(this);
alert(a);

}
e() //字符串a
上面的示例说明,如果在函数体内使用call和apply方法动态调用外部函数,并把call和apply方法的第一个参数值设置为关键字this,那么当前函数e将继承函数f的所有成员。使用call和apply方法能够复制调用函数的内部变量给当前函数体,更改了函数f的内部关键字this指向函数e,这样函数e就可以引用函数f的内部成员。
最后,再看一个比较复杂的示例。在这个示例中将演示如何使用apply方法循环更改当前指针,从而实现循环更改函数的结构。
function r( x ){

return ( x );

}
function f( x ){

x[0] = x[0] + ">";
return x;

}
function o(){

var temp = r;
r = function(){
    return temp.apply( this, f( arguments ) );
}

}
function a(){

o();
alert( r( "=" ) );

}
for( var i = 0 ; i < 10; i ++ ){

a();

}
执行上面代码后会看到,提示信息框中的提示信息不断变化。该示例的核心就在于函数o的设计。在这个函数中,首先使用一个临时变量存储函数r。然后修改函数r的结构,在修改的函数r的结构中,通过调用apply方法修改原来函数r的指针指向当前对象,同时执行原函数r,并把执行函数f的值传递给它,从而实现修改函数r的return语句的后半部分信息,即为返回值增加一个前缀字符“=”。这样每次调用函数o时,都会为其增加一个前缀字符“=”,从而形成一种动态的变化效果。
当然,call和apply方法的应用是非常灵活的,在大型JavaScript技术框架中经常会用到它们,利用它们可以实现动态更改对象的指针,从而实现各种复杂的功能。

相关文章
|
4月前
|
存储 JavaScript 前端开发
JavaScript编程实现tab选项卡切换的效果+1
JavaScript编程实现tab选项卡切换的效果+1
|
5月前
|
JavaScript 前端开发 编译器
解锁JavaScript模块化编程新纪元:从CommonJS的基石到ES Modules的飞跃,探索代码组织的艺术与科学
【8月更文挑战第27天】随着Web应用复杂度的提升,JavaScript模块化编程变得至关重要,它能有效降低代码耦合度并提高项目可维护性及扩展性。从CommonJS到ES Modules,模块化标准经历了显著的发展。CommonJS最初专为服务器端设计,通过`require()`同步加载模块。而ES Modules作为官方标准,支持异步加载,更适合浏览器环境,并且能够进行静态分析以优化性能。这两种标准各有特色,但ES Modules凭借其更广泛的跨平台兼容性和现代语法逐渐成为主流。这一演进不仅标志着JavaScript模块化的成熟,也反映了整个JavaScript生态系统的不断完善。
59 3
|
2月前
|
前端开发 JavaScript 持续交付
提高JavaScript编程效率
提高JavaScript编程效率
31 3
|
2月前
|
自然语言处理 JavaScript 前端开发
JavaScript闭包:解锁编程潜能,释放你的创造力
【10月更文挑战第25天】本文深入探讨了JavaScript中的闭包,包括其基本概念、创建方法和实践应用。闭包允许函数访问其定义时的作用域链,常用于数据封装、函数柯里化和模块化编程。文章还提供了闭包的最佳实践,帮助读者更好地理解和使用这一强大特性。
24 2
|
4月前
|
JavaScript 前端开发
JavaScript编程实现tab选项卡切换的效果
JavaScript编程实现tab选项卡切换的效果
|
4月前
|
JavaScript 前端开发
用JavaScript编程控制网页上checkbox选择状态:全选、全部取消、反选
用JavaScript编程控制网页上checkbox选择状态:全选、全部取消、反选
|
4月前
|
JavaScript 前端开发 安全
JavaScript编程实现字符和字符串翻转
JavaScript编程实现字符和字符串翻转
|
4月前
|
JavaScript 前端开发
用JavaScript编程定义二维数组并初始化,然后输出元素值
用JavaScript编程定义二维数组并初始化,然后输出元素值
|
5月前
|
JavaScript 前端开发 安全
揭秘TypeScript的魔力:它是如何华丽变身为JavaScript的超能英雄,让您的代码飞入全新的编程维度!
【8月更文挑战第22天】在Web开发领域,JavaScript是最主流的编程语言之一。但随着应用规模的增长,其类型安全和模块化的不足逐渐显现。为解决这些问题,微软推出了TypeScript,这是JavaScript的一个超集,通过添加静态类型检查来提升开发效率。TypeScript兼容所有JavaScript代码,并引入类型注解功能。
43 2
|
5月前
|
JavaScript 前端开发 Oracle