js的apply和call函数"源码"以及执行性能

二叶草 2020年2月14日21:48:18函数代码评论阅读模式
Function.prototype.apply=function(x){
    x = x || {};//防止this为null
    x['fn'] = this;
    var list = arguments[1] || [];
    var args = '';//参数列表
    var type;
    for (var i = 0; i<list.length;i++) {        
        if (typeof list[i] === 'string') {//参数是字符串,需要在两边加引号,因为在拼接参数的时候会被丢掉
            type = '"'+list[i] + '"';
        }
        else if (typeof list[i] === 'function') {//参数是函数的话,"反编译"出函数的代码
            type = list[i].toString();
        }
        else if (typeof list[i]  === 'object' ) {
            if (/function Array()/.test(list[i]['constructor'])) {//参数是数组则在数组两边加上[,]
                type = "[" + list[i] + "]";
            } else {
                type = JSON.stringify(list[i])//对象的话,json化,然后执行的时候反json化
            }   
        } else {
            type = list[i];//数字类型
        }
        args = args + type +',';
    }
    args = args.slice(0,args.length-1);//去掉最后的逗号
    var q = new Function('var x = arguments[0];x.fn('+args+')')//new Function的时候作用域是独立的,无法访问call里面的x,所以需要传进去
    q(x)
}
function parse_Array_Object(arg) {
    //可以npm install traverse-deep和结合js的map函数进行相应处理,主要是遍历arg的每一个元素,进行类型判断,然后返回相应的字符串
    //to do ..
}

function a(a){
    console.log(this,arguments)
}
a.apply(null,[1,2])
暂时不支持复杂的变量类型。。。

Function.prototype.call=function(x){
    x = x || {};
    x['fn'] = this;
    var args = '';//参数列表
    var type;
    for (var i = 1; i<arguments.length;i++) {       
        if (typeof arguments[i] === 'string') {//参数是字符串,需要在两边加引号,因为在拼接参数的时候会被丢掉
            type = '"'+arguments[i] + '"';
        }
        else if (typeof arguments[i] === 'function') {//参数是函数的话,"反编译"出函数的代码
            type = arguments[i].toString();
        }
        else if (typeof arguments[i]  === 'object' ) {//数组和对象可能含有复杂的组合类型数据,可以通过遍历变量转成字符串
            if (/function Array()/.test(arguments[i]['constructor'])) {//参数是数组则在数组两边加上[,]
                type = "[" + arguments[i] + "]";
            } else {
                type = JSON.stringify(arguments[i])//对象的话,json化,然后执行的时候反json化
            }   
        } else {
            type = arguments[i];//数字类型
        }
        args = args + type +',';
    }
    args = args.slice(0,args.length-1);//去掉最后的逗号
    var q = new Function('var x = arguments[0];x.fn('+args+')')//new Function的时候作用域是独立的,无法访问call里面的x,所以需要传进去
    q(x)
}

function parse_Array_Object(arg) {
    //可以npm install traverse-deep和结合js的map函数进行相应处理,主要是遍历arg的每一个元素,进行类型判断,然后返回相应的字符串
    //to do ..
}
function a(a){
    console.log(this,arguments)
}
a.call({d:1},2,'s',[1,2,3],{s:1},function() {})
console.time('start');
function a() {}
for (var i=0; i<100000000; i++) {
  a();
}
console.timeEnd('start');
console.time('start');
function a() {}
for (var i=0; i<100000000; i++) {
  a.call();
}
console.timeEnd('start');
console.time('start');
function a() {}
for (var i=0; i<100000000; i++) {
  a.apply();
}
console.timeEnd('start');

执行后会发现,用apply或call执行一个函数所需的时间是直接执行的10倍左右,apply和call需要更多的时间开销。

本文来源于:js的apply和call函数"源码"以及执行性能-变化吧门户
特别声明:以上文章内容仅代表作者本人观点,不代表变化吧门户观点或立场。如有关于作品内容、版权或其它问题请于作品发表后的30日内与变化吧联系。

  • 赞助本站
  • 微信扫一扫
  • weinxin
  • 加入Q群
  • QQ扫一扫
  • weinxin
二叶草
Go语言中的常量 函数代码

Go语言中的常量

1 概述 常量,一经定义不可更改的量。功能角度看,当出现不需要被更改的数据时,应该使用常量进行存储,例如圆周率。从语法的角度看,使用常量可以保证数据,在整个运行期间内,不会被更改。例如当前处理器的架构...
Go语言的接口 函数代码

Go语言的接口

Go语言-接口 在Go语言中,一个接口类型总是代表着某一种类型(即所有实现它的类型)的行为。一个接口类型的声明通常会包含关键字type、类型名称、关键字interface以及由花括号包裹的若干方法声明...
Go语言支持的正则语法 函数代码

Go语言支持的正则语法

1 字符 语法 说明 . 任意字符,在单行模式(s标志)下,也可以匹配换行 字符类 否定字符类 d Perl 字符类 D 否定 Perl 字符类 ASCII 字符类 否定 ASCII 字符类 pN U...
Go语言的包管理 函数代码

Go语言的包管理

1 概述 Go 语言的源码复用建立在包(package)基础之上。包通过 package, import, GOPATH 操作完成。 2 main包 Go 语言的入口 main() 函数所在的包(pa...

发表评论