JavaScriptの関数を深く理解する

黄舟
黄舟オリジナル
2017-03-02 14:56:271347ブラウズ

この記事は、Web 開発者が知っておくべきすべての JavaScript 関数の基本的な知識を提供することを目的としています。

ソフトウェア開発者にとって関数は空想の世界ではありません。日常業務に少しでもコーディングが含まれる場合、最終的には 1 つ以上の関数を作成/変更しているはずです。

要するに、関数は特定の操作を実行するステートメントのセットにすぎません。関数にはいくつかの入力パラメータ (関数本体で使用) があり、実行後に値を返す場合があります。

JavaScript 関数にもこれらのプロパティがありますが、それらは単なる通常の関数ではありません。 JavaScript 関数はオブジェクトです。私がかつて JavaScript オブジェクトについて書いた記事をチェックしてください。そこでは、JavaScript のほとんどすべてがオブジェクトであると述べました。

オブジェクトとして、JavaScript 関数はプロパティと他の関数 (メソッド) を持つことができます。 JavaScript での典型的な関数定義を見てみましょう。

そうです

そうです。上記の機能には大掛かりなものはなく、単にブログへの訪問者を歓迎するだけです。ただし、これは JavaScript 関数がどのようなものかを示しています。関数定義はキーワード function で始まり、その後に関数名、空の関数、またはパラメータを含む括弧が続きます。実際の関数コード (JavaScript ステートメント) は、中括弧 { } で囲まれています。関数の場合、return ステートメントはオプションです。 JavaScript 関数は常に値を返します。 function 本体に return ステートメントがない場合、function は unknown を返します。 function开始,然后是函数名,空的或有参数的括号。实际的函数代码(JavaScript语句)被封装在一对花括号内{ }。对于函数而言,return语句是可选的。JavaScript函数总是会返回一个值。当function主体中没有return语句时,那么function返回undefined。

下面的代码调用传递visitor name作为参数的函数。

function myNotSoGreatFunc(visitor) {
   console.log("Welcome to Code Morning Mr. " + visitor);
}

到现在为止,我们了解了函数非常基本的特征。现在,我们将对JavaScript函数的一些高级概念一探究竟。

匿名函数

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

myNotSoGreatFunc("Bob Martin");
// Output: 
// Welcome to Code Morning Mr. Bob Martin.

上述语法被也被称为函数表达式。你可以把变量addNumbers 当作函数名,以及像下面这样调用该函数。

var addNumbers = function (x, y) { return x + y; }

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

var sum = addNumbers(2, 3);

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

我可以通过传递任意函数作为第一个参数来调用函数calculate。

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);
}

你可以看到将函数作为参数传递是多么容易。这种模式在AJAX中大量使用,当你在AJAX调用完成后,传递回调函数处理成功或失败的场景时。

关于参数的更多内容

JavaScript是非常灵活的,当涉及到传递或访问函数参数的时候。让我们看一下函数参数可以被操纵的方式。

缺少参数

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

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

Arguments对象

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

function callMe(a, b, c) {
   console.log("c is " + typeof c);
}

callMe("Code", "Morning"); 
// Output: "c is undefined"
callMe("Learn", "JavaScript", "Functions"); 
// Output: "c is string"

此函数假设没有传递任何参数,但就像我说的,你可以传递任何数量的参数到JavaScript函数。我可以像这样调用这个函数:

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

每个参数可以从arguments对象作为一个数组项被访问。被传递给函数的arguments的总数可从arguments.length属性获得。

默认参数

你是C ++或C#程序员吗?你见过使用默认参数的函数吗?也许你会回答yes! ECMAScript 6带来了JavaScript的这一特性,就是你可以定义带有默认参数的函数。

callMe("Code", "Morning", "Mr. Programmer");
// Output":
// Code
// Morning
// Mr. Programmer
// Total arguments passed: 3

该函数有礼貌地地迎接了博客访问者。它有两个参数name 和profession

次のコードは、訪問者名をパラメーターとして渡す関数を呼び出します。

function greetMyVisitors(name, profession = "The cool programmer") {
    alert("Welcome Mr. " + name + ", " + profession);
}

ここまでで、関数の非常に基本的な特性を理解しました。ここで、JavaScript 関数の高度な概念をいくつか見ていきます。 🎜🎜匿名関数🎜🎜JavaScript 関数は匿名にすることができます。これは、関数宣言から関数名を省略できることを意味します。ただし、関数は変数に格納する必要があります。 🎜
greetMyVisitors("Justin Bieber", "The singer"); 
// Shows the message "Welcome Mr. Justin Bieber, The singer"

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

greetMyVisitors("John Papa", undefined); 
// Shows the message "Welcome Mr. John Papa, The cool programmer"
🎜上記の構文は関数式とも呼ばれます。変数 addNumbers を関数名として使用し、以下のように関数を呼び出すことができます。 🎜
function wakeUpAndCode() {
   function wakeUp() {
      console.log("I just woke up");
   }

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

   wakeUp();
   code();
}

wakeUpAndCode();

// Output:
// I just woke up
// I am ready to code now
🎜 関数式は、関数をパラメータとして別の関数に渡したいときに非常に便利です。簡単な例を使ってこれを理解してみましょう。 🎜
(function() {
   // Your awesome code here
}());
🎜まず、2 つの匿名関数を作成しました。最初の関数は 2 つの数値の加算を返し、2 つ目は 2 つの数値の乗算を返します。非常にシンプルで、誇示するものは何もありません。次に、関数 calculate を定義します。この関数は、最初の引数として関数を受け取り、その後に 2 つの数値を受け取る 2 つの引数を受け取ります。 🎜🎜最初の引数として任意の関数を渡すことで、関数 Calculate を呼び出すことができます。 🎜
(function() {
   console.log("I run on my own.");
}());
🎜関数をパラメータとして渡すことがいかに簡単であるかがわかります。このパターンは、AJAX 呼び出しの完了後に成功または失敗のシナリオを処理するコールバック関数を渡すときに、AJAX で頻繁に使用されます。 🎜🎜パラメータの詳細🎜🎜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&#39;m a programmer, I can only do Gangnam style..");
   }

   this.canWriteJavaScript = function() {
      return expertise === "JavaScript";
   }
}

Arguments オブジェクト

🎜すべての JavaScript 関数には、arguments と呼ばれる特別なオブジェクトがあり、これは関数呼び出し中に渡されるパラメーター配列です。このオブジェクトを使用して、個々のパラメーターにアクセスしたり、関数に渡されたパラメーターの合計数を取得したりできます。 🎜
var javaProgrammer = new Programmer("Mohit Srivastava", "Infosys", "Java");
var dotnetProgrammer = new Programmer("Atul Mishra", "Prowareness", ".NET");
🎜この関数はパラメーターが渡されないことを前提としていますが、前述したように、JavaScript 関数には任意の数のパラメーターを渡すことができます。この関数は次のように呼び出すことができます: 🎜
var jsProgrammer = Programmer("Douglas Crockford", "Yahoo", "JavaScript")
🎜 各引数は、arguments オブジェクトから配列項目としてアクセスできます。関数に渡される arguments の合計数は、arguments.length プロパティから取得できます。 🎜

デフォルト パラメータ

🎜 あなたは C++ または C#プログラマですか?デフォルトのパラメータを使用する関数を見たことがありますか?たぶん、あなたは「はい」と答えるでしょう! ECMAScript 6 は、JavaScript のこの機能をもたらします。つまり、デフォルトのパラメーターを使用して関数を定義できます。 🎜
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..");
   }
}
🎜ブログ訪問者を丁寧に迎える機能です。これには 2 つのパラメータ nameprofession があり、メッセージ ボックスにウェルカム メッセージが表示されます。呼び出し中に引数が渡されない (または「未定義」) 場合、2 番目の引数はデフォルト値を受け取ります。 🎜
&#39;use strict&#39;
 function doSomething() { ... }
 ....
 ....
🎜ネストされた関数🎜🎜 関数には、その中に 1 つ以上の関数を含めることができます。内部関数には内部に再度関数を含めることができます。以下の操作を見てみましょう。 🎜
function wakeUpAndCode() {
   function wakeUp() {
      console.log("I just woke up");
   }

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

   wakeUp();
   code();
}

wakeUpAndCode();

// Output:
// I just woke up
// I am ready to code now

函数wakeUpAndCode包含两个内部函数wakeUp和code。当调用wakeUpAndCode时,函数主体开始执行函数主体。在外部函数中只有两个可执行语句,调用wakeUpcode的方法。调用wakeUp将执行内部wakeUp函数,这将写入string“I just woke up”到控制台。调用code将会写入“I am ready to code now”string到控制台。

内部函数可以访问所有外部函数的变量和参数。内部函数是函数内部某种private实现,并且不能从外部函数以外被调用。内部函数的使用生成了JavaScript闭包,这个我将另起一篇文章讨论。

立即执行函数表达式(IIFE,发音iffy)

IIFE是被立即调用执行的匿名函数表达式。IIFE看上去像这样:

(function() {
   // Your awesome code here
}());

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

IIFE块中定义的任何变量或函数对块而言是本地的,并且不能被这个范围以外的任何代码改变。

看看IIFE的这个例子。此函数没有调用也会自动执行。

(function() {
   console.log("I run on my own.");
}());

只需在plunker中复制并粘贴代码,看看在浏览器控制台中的输出。如果你不知道去哪里找浏览器控制台,那么只要在浏览器窗口中按下F12就会出现开发者工具。跳转console选项卡以查看console.log语句的所有输出。

IIFE是一个在代码中创建局部范围的很好方法。它们可以帮助你保护变量和函数,以避免被应用程序的其他部分更改或覆盖。JavaScript中IIFE的其他优势?它们是如何解决全局范围污染问题的?欢迎点击查看我关于立即执行函数表达式的文章。

构造函数

函数可以充当构造器的角色,并且可以使用构造函数来创建新的对象。这是使JavaScript面向对象的特点之一。使用构造函数的好处是,你将能够通过预定义的属性和方法,创造尽可能多的对象。如果你由此关联到其他语言中的类和对象,那么你做的对。

让我们创建一个带有一些属性和方法的构造函数Programmer。你可以假设它在你最喜欢的语言中是一个类。

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&#39;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");

虽然也可以创建一个使用对象文本语法带有相同属性和方法的对象,但我们需要多次编写相同的代码,这可不是什么伟大的实践。如果你知道编程DRY原则,那么你就不会不赞同我。构造函数使得可以一次定义对象,并创建真正的实例,无论什么时候你想要。

警告!

始终使用new关键字来从构造器创建新的对象。忘记了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模式下运行代码是一个良好的实践。

&#39;use strict&#39;
 function doSomething() { ... }
 ....
 ....

在这篇文章中,我几乎已经涵盖了有关函数的所有内容。函数被认为是JavaScript中的一等公民。理解函数可能是最重要的事情,如果你想掌握JavaScript的话。

欢迎各位指正。

上記は JavaScript の関数をより深く理解するための内容です。さらに関連する内容については、PHP 中国語 Web サイト (www.php.cn) に注目してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。