JavaScript中的函数

定义函数 - 函数声明和函数表达式

函数声明

// function关键字,functionName函数名,arg参数(可选)
function functionName(arg){
    //函数体
}

注意

  • 当函数的参数是一个值,若被调用函数改变了这个参数的值,这样的改变不会影响到全局或调用的函数。

  • 但当函数的参数是一个对象(即一个非原始值,例如Array或用户自定义的其它对象),若函数改变了这个对象的属性,这样的改变对函数外部是可见的。

函数表达式

//var关键字,variableName变量名,表达式赋值等号,声明变量后的分号,functionName函数名(可选,没有的话叫做匿名函数)
var variableName = function functionName(arg){
    //函数体
};

函数声明和函数表达式的区别

解析器会率先读取函数声明,使其在执行任何代码之前就可以访问(也就是 函数声明提升);
而函数表达式则需要解析器执行到它所在的代码行才会被解释执行。

匿名函数

  • JavaScript函数可以是匿名的。这意味着你可以从函数声明中省略函数名。但是,函数必须存储在变量中。

  • var addNumbers = function (x, y) { return x + y; }上述语法被也被称为函数表达式。

  • 你可以把变量addNumbers 当作函数名,以及像下面这样调用该函数。var sum = addNumbers(2, 3);

  • 当你想传递一个函数作为参数给另一个函数时,函数表达式就非常方便了。让我们用一个简单的例子来试着了解这一点。

var add = function (first, second) { return first + second };
var multiply = function (first, second) { return first * second };
function calculate(fun, a, b) {   
return fun(a, b);
}

首先我已经创建了两个匿名函数。第一个返回两个数的加法运算,第二个返回两个数的乘法运算。相当简单,没有什么可值得炫耀的地方。然后,我定义函数calculate,这个函数接受函数作为第一个参数后跟两个参数接受两个数字。我可以通过传递任意函数作为第一个参数来调用函数calculate。

var sum = calculate(add, 2, 3); // sum = 5
var multiplication = calculate(multiply, 2, 3); // multiplication = 6

关于参数

缺少参数

调用函数时,函数的参数数量可以比要求的更少或更多。如果你调用的函数的参数比声明的少,那么缺少的参数被设置为undefined。

function callMe(a, b, c) {
   console.log("c is " + typeof c);
}
callMe("Code", "Morning"); // c is undefined
callMe("Learn", "JavaScript", "Functions"); //  c is string

Arguments对象

所有的JavaScript函数有一个特殊的对象,叫做arguments,它是在函数调用过程中传递的参数数组。该对象可以被用来访问单个参数或获得传递到函数的参数总数。

function callMe() {
   var i;  
   for (i = 0; i < arguments.length; i++) {     
   console.log(arguments[i]);
   }
   console.log("Total arguments passed: " + arguments.length);
}

此函数假设没有传递任何参数,但就像我说的,你可以传递任何数量的参数到JavaScript函数。我可以像这样调用这个函数:callMe("Code", "Morning", "Mr. Programmer");// Code Morning Mr.Programmer Total arguments passed: 3每个参数可以从arguments对象作为一个数组项被访问。被传递给函数的arguments的总数可从arguments.length属性获得。

默认参数

function greetMyVisitors(name, profession = "The cool programmer") {
    alert("Welcome Mr. " + name + ", " + profession);
}
//该函数有礼貌地地迎接了博客访问者。它有两个参数name 和profession,并在消息框中显示一个欢迎消息。
//如果在调用过程中没有参数(或“undefined”)传递,那么第二个参数取用默认值。

greetMyVisitors("Justin Bieber", "The singer"); 
//Welcome Mr. Justin Bieber, The singer

greetMyVisitors("Bob Martin"); 
// Welcome Mr. Bob Martin, The cool programmer

greetMyVisitors("John Papa", undefined); 
// Welcome Mr. John Papa, The cool programmer

嵌套函数

函数可以在它的内部包含一个或多个函数。内部函数可能会在内部再次包含函数。

function wakeUpAndCode() {   

function wakeUp() {     
console.log("I just woke up");
   }   

function code() {    
 console.log("I am ready to code now");
   }

   wakeUp();
   code();
}
wakeUpAndCode();//  I just woke up   I am ready to code now

函数wakeUpAndCode包含两个内部函数wakeUpcode。当调用wakeUpAndCode时,函数主体开始执行函数主体。在外部函数中只有两个可执行语句,调用wakeUpcode的方法。调用wakeUp将执行内部wakeUp函数,这将写入I just woke up到控制台。调用code将会写入I am ready to code now到控制台。内部函数可以访问所有外部函数的变量和参数。内部函数是函数内部某种private实现,并且不能从外部函数以外被调用。

立即执行函数表达式

(function() { 
  // Your code here
}());
  • 所有你要做的就是创建一个匿名函数,在函数定义后马上放一对圆括号以调用函数,最后将所有代码封装在另一对圆括号中。最外层的括号将它里面的所有一切转变成一个表达式,因为括号不能包含JavaScript语句。函数定义后面的圆括号则立即调用函数。

  • 这其中定义的任何变量或函数不能被这个范围以外的任何代码改变。

  • 这是一个在代码中创建局部范围的很好方法。它们可以帮助你保护变量和函数,以避免被应用程序的其他部分更改或覆盖。

构造函数

函数可以充当构造器的角色,并且可以使用构造函数来创建新的对象。这是JavaScript面向对象的特点之一。使用构造函数的好处是,你将能够通过预定义的属性和方法,创造尽可能多的对象。

function Programmer(name, company, expertise) {  
this.name = name;   
this.company = company;  
this.expertise = expertise;  
this.writeCode = function() {     
console.log("Writing some public static thing..");
   }   
this.makeSkypeCall = function() {    
  console.log("Making skype call..");
   }  
this.doSalsa = function() {    
  console.log("I'm a programmer, I can only do Gangnam style..");
   }  
this.canWriteJavaScript = function() {     
  return expertise === "JavaScript";
   }
}

函数有三个参数,并创建了一个具有三个属性和四种方法的对象。
var javaProgrammer = new Programmer("Mohit Srivastava", "Infosys", "Java”); var dotnetProgrammer = new Programmer("Atul Mishra", "Prowareness", ".NET");
虽然也可以创建一个使用对象文本语法带有相同属性和方法的对象,但我们需要多次编写相同的代码。构造函数使得可以一次定义对象,并创建真正的实例。

注意

始终使用new关键字来从构造器创建新的对象。
  • var jsProgrammer = Programmer("Douglas Crockford", "Yahoo", "JavaScript")
    最终将添加所有属性和方法到全局的window对象,原因是,除非明确指定,否则“this”指向全局的window对象。使用new 设置“this”上下文到被创建的当前对象。

  • 然而,有一种变通方法可以来克服这个问题。你可以改变构造函数的实现以使域安全,然后在创建新的对象时,你就可以愉快地忽略new 关键字了。请参见以下修改了的构造函数代码。

function Programmer(name, company, expertise) {  
if(!(this instanceof Programmer)) {     
    return new Programmer(name, company, expertise);
   }  
this.name = name; 
this.company = company;   
this.expertise = expertise;   
this.writeCode = function() {     
console.log("Writing some public static thing..");
   }
}

if 条件检查了this 对象是否是Programmer的一个实例。如果不是,它会创建一个新的Programmer对象,并通过再次调用构造器返回相同的内容。

注意:你无法在不使用’new’关键字的情况下,在Strict模式下从构造器创建一个新的对象。Strict模式强制一些编码准则,并且在你写的东西不安全的情况下会抛出错误。要启用Strict模式,你只需要添加在你的代码开头添加字符串 ‘use strict’。在Strict模式下运行代码是一个良好的实践。

'use strict'
 function doSomething() { ... }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,921评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,635评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,393评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,836评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,833评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,685评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,043评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,694评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,671评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,670评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,779评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,424评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,027评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,984评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,214评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,108评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,517评论 2 343

推荐阅读更多精彩内容

  • 函数和对象 1、函数 1.1 函数概述 函数对于任何一门语言来说都是核心的概念。通过函数可以封装任意多条语句,而且...
    道无虚阅读 4,521评论 0 5
  • 查阅书籍:JavaScript权威指南 函数声明与函数表达式 用函数声明定义的函数,函数可以在函数声明之前调用,而...
    雪萌萌萌阅读 348评论 1 9
  • 第5章 引用类型(返回首页) 本章内容 使用对象 创建并操作数组 理解基本的JavaScript类型 使用基本类型...
    大学一百阅读 3,211评论 0 4
  • 第2章 基本语法 2.1 概述 基本句法和变量 语句 JavaScript程序的执行单位为行(line),也就是一...
    悟名先生阅读 4,114评论 0 13
  • 一、MACD指标对0轴的突破 1.MACD指标向上突破0轴 当DIFF线由下向上突破0轴,表明多头行情即将悄然形成...
    chiang1219阅读 241评论 0 2