javascript 函数

javaScript function

一、形参和实参

1.可选形参

//给参数赋初值做到形参可选
function f(a,b){
    b = b || 0;  //给参数b赋初值
    return a+b;
}
//ES6写法
function f(a=1,b=0){
    return a+b;
}
f(); //1

2.函数参数优化

函数参数的优化的目的是使函数在调用时更方便,运行时更健壮

//针对函数参数过多记不住参数顺序优化,
//使用对象属性,通过键值对传入参数
function easyCopy(args){
     copy(args.from, args.from_start=0, args.to,args.to_start=0,args.length)
}
let a = [1,2,3,4],b=[];
easyCopy({to:b,from:a,length:a.length});
//以上函数完成了从a到b的数组拷贝,实参传递时使用对象属性形式,不必关心参数的顺序

//运行时检测实参类型,防止非法实参在程序运行时报错
//以下求和函数返回数组或类数组元素的累加和
function sum(a){
    if(isArrayLike(a)){    //isArrayLike检测参数是否为类数组
        let total = 0;
        for(let i = 0;i<a.length;i++){
            let element = a[i];
            if(element == null) continue;//跳过null和undefined
            if(isFinite(element)) total += element;
            else throw new Error('elements must be finite numbers');
        }
        return total;
    }
    else throw new Error('arguments must be array-like');
}

3.arguments

  • arguments是一个类数组实参对象,拥有length和数字索引,没有数组方法
function f(x,y){
    return arguments[0]+arguments[1]+arguments.length;
}
  • arguments.callee
    指向正在执行的函数,在递归函数中很有用,需要注意的是arguments.callee在严格模式中不支持
function factorial(n){
    if(n == 1) return n;
    return n * arguments.callee(n-1);
}
factorial(5);  //120

//使用arguments.callee有一个好处,不受函数名改变的限制,
//可以实现匿名函数递归,还能防止函数名改变出错,看以下例子
function factorial1(n){
    if(n == 1) return n;
    return n * factorial1(n-1);
}
let f = factorial1;
factorial1 = null;
f(5); // Uncaught TypeError: factorial1 is not a function

//使用arguments.callee实现递归则不会出现此问题
let fun = factorial;
factorial = null;
fun(5); //120
  • arguments.caller
    指代调用当前正在执行函数的函数,通过caller可以访问调用栈,同样,严格模式不支持

二、函数调用

函数调用最需要关心的是this的指向,方法和this关键字是面向对象编程的核心

1.作为普通函数调用

作为普通函数调用this指向全局对象(非严格模式)或者undefined(严格模式)

function f(){
    return this; 
}
f();//window

'use strict'
function f1(){
  return this; 
}
f1();//undefined

2.作为对象方法调用

  • this指向调用该函数的对象
let obj = {
    name : 'Henry',
    getName : function(){
        return this.name;
    }
}
obj.getName(); //'Henry'
  • 对象里的嵌套函数
    作为函数调用指向全局或undefined,作为方法调用指向调用它的函数
var o = {
    checkThis: function(){
        let self = this; //将this保存至一个变量
        console.log(this == o); //true,this指代对象o
        f();  
        function f(){              //定义对象内嵌套函数
            console.log(this);     //window
            console.log(self == o);//true
        }
    }
}

3.作为构造函数调用

this指向返回的对象

let myClass = function(){
    this.name = 'Henry';
}
let obj = new myClass();
console.log(obj.name);//'Henry'

4.通过call和apply调用

this指向第一个参数,第一个参数传入null指向window(非严格模式)或undefined(严格模式)

let name = 'window'
function getName(){
    return this.name;
}
let obj = {
    name : 'Henry',
}
getName(); //'window'
getName.call(obj); //'Henry',this指向obj, this.name相当于obj.name
getName.apply(obj); //'Henry'

5.丢失的this

看下面的例子

let obj = {
    name : 'Henry',
    getName : function(){
        return this.name;
    }
}
obj.getName();//'Henry',this指向obj
let getName2 = obj.getName;
window.name = 'window'
getName2(); //'window',此时相当于普通函数调用,this指向window对象
//当对象的方法赋值给一个普通函数时this指向改变,不再指向原对象

三、闭包

理解闭包最重要的是理解变量的作用域和作用域链

1.变量的作用域及作用域链

  • 函数内变量的作用域是在函数定义时决定的,不是函数调用时决定的
  • 函数内声明的变量是局部变量,会覆同名的全局变量
  • 函数的作用域链可以理解为一个对象列表,每次函数调用都会为之创建一个新的对象用来保存局部变量,把这个对象添加至作用域链当中,当函数中搜索一个变量时,如果函数内部没有声明此变量,会随着作用域链往外层搜索,一直到全局
 var scope = 'global';
function checkScope(){
    let scope = 'local';
    function f(){return scope};
    return f;
}
checkScope()();  //'local'

此例在外部调用了嵌套函数,checkScope没有直接返回嵌套函数执行结果,而是返回了函数对象。
f函数执行时用到了作用域链,这个作用域链是在函数f定义时创建的,函数f的定义在checkScope内部,所以作用域链的第一级是checkScope函数内部,第二级是window,首先搜索到的scope变量是'local'

2.变量的生命周期

  • 全局变量的生命周期是永久的,除非主动删除
  • 函数内部局部变量的生命周期在函数调用完成后结束
  • 闭包内的局部变量在函数调用结束后生命被延续
function func(){
    let a = 1;
    return function(){
        a++;
        return a;
    }
}
let f = func();
f();//2
f();//3
f();//4

当执行let f = func();时,f获得了一个匿名函数的引用,它可以访问到func()被调用时产生的环境,局部变量a一直在这个环境中。既然局部变量所在的环境还能被外界访问,这个局部变量就有了不被销毁的理由。在这个闭包结构中,局部变量的生命被延续了。

四、函数的属性,方法

1.属性

函数定义时参数的个数,arguments.callee.length,可以理解为期望传入的实参个数,不是arguments.length的length,arguments.length是实际传入的实参个数

//此函数使用arguments.callee,不能在严格模式下运行
function check(args){
    let actual = args.length;        //实参的真实个数
    var expected = args.callee.length; //期望实参个数
    if(actual != expected){
        throw Error("Expected" + expected + 'args;got' + actual);
    }
}
function(x,y,z){
    check(arguments); //检查实参个数与期望个数是否一致
    /**函数逻辑*/
}

2.函数方法

  • call/apply

call和apply的第一个参数是要调用的函数的母对象,指定this指向,apply的第二个参数是数组形式的参数对象,call其实是apply的一颗语法糖,后面的参数以单个实参的方式给出

call和apply的用途
(1)改变this指向

var name = 'window';
let obj = {
    name : 'Henry'
}
function getName(){
    return this.name;
}
geName(); //'window'
getName.call(obj); //'Henry'

(2)借用其他对象的方法

function testCall(){
    Array.prototype.push.call(arguments,4,5); 
    return Array.prototype.map.call(arguments,value=>return value+1;);
}
testCall(1,2,3);//[2,3,4,5,6]
//arguments是类数组没有数组方法,但是可以借用数组的方法,
//testCall函数里arguments两次借用数组方法,
//第一次给arguments元素尾部加上4,5;
//第二次用map方法给arguments所有元素加1

(3)模拟bind

Function.prototype.bind = function(context){
    var self = this; //保存原函数
    return function(){
        return self.apply(context, arguments);
        //执行时将传入的context当成新函数体内的this
    }
}
  • bind
    指定函数内部this指向
var obj = {
    name:'Henry'
};
var func = function(){
      console.log(this.name)
}.bind(obj); 
func();//Henry,func函数的内部指向被绑定到了对象obj
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,029评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,395评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,570评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,535评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,650评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,850评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,006评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,747评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,207评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,536评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,683评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,342评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,964评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,772评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,004评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,401评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,566评论 2 349

推荐阅读更多精彩内容

  • 函数函数定义与调用变量作用域全局变量方法高阶函数闭包箭头函数$generator$ 函数 函数定义与调用 定义函数...
    染微言阅读 578评论 0 5
  • 本文是大神廖雪峰的JavaScript教程学习笔记。并不是教程,如有需要,请前往廖雪峰大神大博客. 一、函数定义和...
    0o冻僵的企鹅o0阅读 485评论 1 3
  • JavaScript笔记(三) 函数 理解函数 Javascript函数的参数与大多数其他语言中的函数的参数不同。...
    运维开发笔记阅读 272评论 0 0
  • 函数就是最基本的一种代码抽象的方式。 定义函数function abs(x) {if (x >=0){return...
    _我和你一样阅读 441评论 0 0
  • 在JavaScript中,函数即对象,程序可以随意操控它们。比如,JavaScript可以把函数赋值给变量,或者作...
    kissLife阅读 905评论 0 0