2021年8月28日

関数

スクリプトの色々な場所で同じアクションを実行する必要がある場合がよくあります。

例えば、訪問者がログイン/ログアウトしたり、また複数の箇所で見栄の良いメッセージを表示する必要があったりします。

関数はプログラムのメインの “構成要素” です。これによりコードを繰り返すことなく何度も呼び出すことができます。

私たちは既に組み込み関数の例を見ています。 alert(message), prompt(message, default)confirm(question)です。これと同じように私たち自身も関数を作ることができます。

関数定義

関数を作るために、関数定義 を使います。

次のようになります:

function showMessage() {
  alert( 'Hello everyone!' );
}

function キーワードが最初にきて、次に 関数名 がきます、そして括弧の中に パラメータ のリスト(カンマ区切り、上の例では空)がきて、最後に中括弧の間に関数のコード、 “関数本体” です。

function name(parameters) {
  ...body...
}

作成した関数はその関数名で呼び出すことができます: showMessage()

例:

function showMessage() {
  alert( 'Hello everyone!' );
}

showMessage();
showMessage();

showMessage() の呼び出しは、関数のコードを実行します。この例では、2度メッセージが表示されます。

この例は関数のメインの目的の1つを明確に示しています: コードの複製を回避する、と言うことです。

もしメッセージ内容、または表示方法を変更する必要がある場合、1箇所のコード(関数)を修正するだけで十分です。

ローカル変数

関数内で定義された変数は、関数内でのみ参照可能です。

例:

function showMessage() {
  let message = "Hello, I'm JavaScript!"; // ローカル変数

  alert( message );
}

showMessage(); // Hello, I'm JavaScript!

alert( message ); // <-- エラー! 変数は関数のローカルです

外部変数

関数は外部変数にアクセスすることもできます。次の例を見てください:

let userName = 'John';

function showMessage() {
  let message = 'Hello, ' + userName;
  alert(message);
}

showMessage(); // Hello, John

関数は外部変数に対してフルアクセス権を持ち、変更することもできます。

例:

let userName = 'John';

function showMessage() {
  userName = "Bob"; // (1) 外部変数の変更

  let message = 'Hello, ' + userName;
  alert(message);
}

alert( userName ); // 関数呼び出しの前は John

showMessage();

alert( userName ); // Bob, 関数によって値が変更されました

外部の変数は、同じ名前のローカル変数が存在しない場合にのみ使われます。そのため、let を忘れた場合、意図せず外部の変数を変更してしまう可能性があります。

同じ名前の変数が関数内に宣言されている場合は、外部変数を 隠します。例えば、以下のコードでは関数はローカルの userName を使います。外部の userName は無視されます。

let userName = 'John';

function showMessage() {
  let userName = "Bob"; // ローカル変数の宣言

  let message = 'Hello, ' + userName; // Bob
  alert(message);
}

// 関数は作られ独自の userName を使います
showMessage();

alert( userName ); // John, 変更されていません。関数は外部変数へアクセスしませんでした
グローバル変数

上のコードにおいて、外部の userName のような、関数の外で宣言されている変数は グローバル と呼ばれます。

グローバル変数はどの関数からも見えます(ローカル変数により隠れていなければ)。

通常、関数は自身のタスクに必要なすべての変数を宣言します。また、グローバル変数にはプロジェクトレベルのデータのみを保持するため、どこからでも見える事が重要です。現代のコードはほとんどもしくは全くグローバル変数を持ちません。ほぼすべての変数は関数に属します。

パラメータ

パラメータを使うことで、任意のデータを関数に渡すことができます。

下の例では、関数は2つのパラメータを持っています: fromtext です。

function showMessage(from, text) { // 引数: from, text
  alert(from + ': ' + text);
}

showMessage('Ann', 'Hello!'); // Ann: Hello! (*)
showMessage('Ann', "What's up?"); // Ann: What's up? (**)

(*)(**) で関数が呼ばれたとき、与えられた値はローカル変数 fromtext にコピーされます。そして関数はそれらを使います。

ここにもう1つ例があります: 私たちは変数 from を持っており、それを関数に渡します。注意してください:関数は常に値のコピーを取得するため、関数の中の処理は from を変更していますが、その変更は外には見えません:

function showMessage(from, text) {

  from = '*' + from + '*'; // "from" をより良く見せる

  alert( from + ': ' + text );
}

let from = "Ann";

showMessage(from, "Hello"); // *Ann*: Hello

// "from" の値は同じで、関数はローカルコピーを変更しています。
alert( from ); // Ann

関数のパラメータとして渡された値は “引数(ひきすう)” とも呼ばれます。 これらの用語を整理すると:

  • パラメータとは、関数の宣言時に括弧内に記述される変数のこと(宣言時の用語)
  • 引数とは、関数が呼び出されたときに渡される値のこと(呼び出し時の用語)

パラメータを列挙して関数を宣言し、引数を渡して関数を呼び出すことになります。

上の例ではこのように言えるでしょう。「関数showMessageは2つのパラメータを持つと宣言されており、from と “Hello” という2つの引数を与えて呼び出されている」

デフォルト値

関数の呼び出し時に引数が与えられていない場合、対応する値は undefined になります。

例えば、前述の関数 showMessage(from, text) は1つの引数で呼ぶことも出来ます:

showMessage("Ann");

それはエラーではありません。このような呼び出しは "*Ann*: undefined" を出力します。text が渡されていないため、textundefined となります。

パラメータのいわゆる “デフォルト” (呼び出し時に省略された場合に使用される)値を、関数宣言の中で = を使用して指定することが可能です:

function showMessage(from, text = "no text given") {
  alert( from + ": " + text );
}

showMessage("Ann"); // Ann: no text given

これで text パラメータが渡されていない場合、 値は "no text given" になります。

ここで、 "no text given" は文字列ですが、より複雑な式にすることもできます。そしてそれはパラメータが無い場合にのみ評価され、代入されます。なので、このようなことも可能です:

function showMessage(from, text = anotherFunction()) {
  // anotherFunction() はテキストが与えられなかった場合にのみ実行されます
  // その結果がtextの値になります
}
デフォルト値の評価

JavaScriptでは、デフォルト値はそれぞれのパラメータが与えられずに関数が呼び出されるたびに評価されます。

上の例だと anotherFunction() は、 text のパラメータが与えられずに showMessage() が呼び出されるたびに実行されます。

代替のデフォルトパラメータ

パラメータのデフォルト値を関数宣言ではなく、後の段階で実行中に設定することが理にかなっている場合があります。

省略されたパラメータをチェックするために、undefined と比較できます:

function showMessage(text) {
  if (text === undefined) {
    text = 'empty message';
  }

  alert(text);
}

showMessage(); // empty message

…もしくは || 演算子:

// パラメータが省略 or  "" の場合, 'empty' を設定
function showMessage(text) {
  text = text || 'empty';
  ...
}

モダンな JavaScript エンジンは NULL合体演算子 ?? をサポートしており、0 などの偽値を通常とみなす場合に適しています:

// count パラメータがない場合は "unknown"
function showCount(count) {
  alert(count ?? "unknown");
}

showCount(0); // 0
showCount(null); // unknown
showCount(); // unknown

値の返却

関数は、実行結果として呼び出しコードに値を戻すことが出来ます。

最もシンプルな例は2つの値の合計を行う関数です:

function sum(a, b) {
  return a + b;
}

let result = sum(1, 2);
alert( result ); // 3

ディレクティブ return は関数の任意の場所に置くことが出来ます。もしも実行がそこに到達したとき、関数は停止し、値を呼び出し元のコードに返します(上の result へ代入します)。

1つの関数に多くの return が出現することもあります。例えば:

function checkAge(age) {
  if (age > 18) {
    return true;
  } else {
    return confirm('Got a permission from the parents?');
  }
}

let age = prompt('How old are you?', 18);

if ( checkAge(age) ) {
  alert( 'Access granted' );
} else {
  alert( 'Access denied' );
}

値なしで return を使うことも出来ます。これは関数を直ぐに終了させます。

例:

function showMovie(age) {
  if ( !checkAge(age) ) {
    return;
  }

  alert( "Showing you the movie" ); // (*)
  // ...
}

上のコードでは、checkAge(age)false を返すと、showMoviealert の処理をしません。

空の return、 または返却がないものは undefined を返します

関数が値を返却しない場合、それは undefined を返却した場合と同じになります。:

function doNothing() { /* empty */ }

alert( doNothing() === undefined ); // true

空の return もまた return undefined と同じです:

function doNothing() {
  return;
}

alert( doNothing() === undefined ); // true
returnと値の間に改行を入れないでください

return が長い式の場合、このように別の行に書くのが魅力的に見えるかもしれません:

return
 (some + long + expression + or + whatever * f(a) + f(b))

JavaScriptは return の後にセミコロンを想定するため、これは動作しません。これは次と同じように動作します:

return;
 (some + long + expression + or + whatever * f(a) + f(b))

従って、これは事実上空の返却になります。なので、値は同じ行に置く必要があります。

もし複数行にまたがった式を返却したい場合は、return と同じ行から開始する必要があります。あるいは、少なくとも次のように開始括弧を置きます:

return (
  some + long + expression
  + or +
  whatever * f(a) + f(b)
  )

これは期待する通りに動作するでしょう。

関数の命名

関数はアクションです。そのため、それらの名前は通常は動詞です。それは簡潔にすべきですが、関数がすることをできるだけ正確に表現してください。そして、コードを読む人が正しい手がかりを得られるようにします。

曖昧なアクションを示す動詞のプレフィックスから関数名を始めることは広く行われています。プレフィックスの意味についてはチーム内での合意が必要です。

例えば、"show" で始まる関数は、通常何かを表示します。

以下で始まる関数…

  • "get…" – 値を返します,
  • "calc…" – 何かを計算します,
  • "create…" – 何かを生成します,
  • "check…" – 何かをチェックし、真偽値を返します, etc

このような名前の例です:

showMessage(..)     // メッセージを表示します
getAge(..)          // 年齢を返します(なんとかしてその値を得る)
calcSum(..)         // 合計を計算し、それを返します
createForm(..)      // フォームを生成します(通常それを返却します)
checkPermission(..) // 権限をチェックし、true/false を返します

決まった位置にプレフィックスを使用すると、関数名を見ただけでそれがどのような種類の処理を行い、どのような値を返すのかを理解することが出来ます。

1つの関数 – 1つのアクション

関数はその名前により提案されたことを正確にするべきです。

通常、2つの独立したアクションは、たとえそれらが一緒に呼ばれるとしても、2つの関数にするのが良いです(その場合は、通常その2つを呼ぶ3つ目の関数を作ります)。

このルールを破るいくつかの例です:

  • getAge – 年齢を取得するとともに 警告 を表示します(取得のみをするべきです)
  • createForm – フォームを作成して、ドキュメントに追加します(作成とその返却だけにするべきです)
  • checkPermissionアクセス許可/拒否 のメッセージを表示するのは良くありません(チェックを実行し、その結果を返すのみにすべきです)

これらの例はプレフィックスの共通の意味を前提としています。これらが意味することは、あなたとあなたのチームで前提を決めるということです。恐らく、コードが異なる振る舞いをするのは普通なことです。しかし、プレフィックスが意味すること、プレフィックスの付いた関数ができること、できないことについてはしっかりとした理解をもっておくべきです。同じプレフィックスの関数はルールに従うべきです。そして、チームはそれを共有するべきです。

究極的に短い関数名

非常に頻繁に 使われる関数は、究極的に短い名前を持っていることがあります。

例えば、jQuery フレームワークは関数 $ を定義しています。LoDash ライブラリは、そのコアな関数として _ を持っています。

それらは例外です。一般的に関数名は簡潔で説明的でなければなりません。

関数 == コメント

関数は短く明確に1つのことを行うべきです。もし関数が大きい場合、恐らくそれを幾つかの小さい関数に分けることは価値があるでしょう。このルールに従うことは簡単ではないこともありますが、間違いなく良いことです。

分割した関数はテストやデバッグが簡単になるだけでなく、 – その存在自体が素晴らしいコメントになります!

例えば、下にある2つの関数 showPrimes(n)を比べてみましょう。どちらも素数n に達するまで出力します。

1つ目のパターンはラベルを使います:

function showPrimes(n) {
  nextPrime: for (let i = 2; i < n; i++) {

    for (let j = 2; j < i; j++) {
      if (i % j == 0) continue nextPrime;
    }

    alert( i ); // a prime
  }
}

2つ目のパターンは、素数の確認をするための追加の関数 isPrime(n) を使います。

function showPrimes(n) {

  for (let i = 2; i < n; i++) {
    if (!isPrime(i)) continue;

    alert(i);  // a prime
  }
}

function isPrime(n) {
  for (let i = 2; i < n; i++) {
    if ( n % i == 0) return false;
  }
  return true;
}

2つ目のパターンのほうが理解しやすいですね。コードの塊の代わりに、アクション(isPrime) の名前を見ます。このようなコードは 自己記述的 と呼ばれる場合があります。

従って、関数はその再利用を意図していない場合でも作ることがあります。それらはコードを構造化し、読みやすくします。

サマリ

関数はこのように定義します:

function name(parameters, delimited, by, comma) {
  /* code */
}
  • パラメータとして関数に渡される値は、ローカル変数にコピーされます。
  • 関数は外部の変数にアクセスすることができます。しかし、それは内側からのみ機能します。関数の外側のコードは、関数のローカル変数を見ることはできません。
  • 関数は値を返すことができます。もしもそれをしなかった場合、戻り値は undefined です。

コードを綺麗で理解しやすいようにするために、その関数内では外部変数ではなく、ローカル変数やパラメータを利用することを推奨します。

パラメータを取得せずに外部変数を変更する関数よりも、パラメータを取得してそれを処理して結果を返す関数の方が、常に理解しやすいものです。

関数名:

  • 名前は、関数がすることを明確に記述するべきです。コードの中で関数呼び出しを見るとき、良い名前であればそれが何をして何を返すのかを簡単に理解することができます。
  • 関数はアクションなので、関数名は通常動詞的です。
  • create…, show…, get…, check… など、数多くのよく知られた関数のプレフィックスが存在します。関数がすることのヒントとしてそれらを使いましょう。

関数はスクリプトの主な構成要素です。今や私たちは基礎をカバーしたので、実際にそれらを作り使い始めることができます。しかし、それはまだほんの始まりに過ぎません。私たちは何度もそれらに戻り、より高度な機能について深めていきます。

タスク

重要性: 4

次の関数は、パラメータ age18 より大きい場合に true を返します。

それ以外の場合には確認を行い、その結果を返します。:

function checkAge(age) {
  if (age > 18) {
    return true;
  } else {
    // ...
    return confirm('Did parents allow you?');
  }
}

もし else が削除された場合、この関数は違う動きになるでしょうか?

function checkAge(age) {
  if (age > 18) {
    return true;
  }
  // ...
  return confirm('Did parents allow you?');
}

これら2つのバリアントの振る舞いで何か違いはあるでしょうか?

違いはありません。

重要性: 4

次の関数は、パラメータ age18 より大きい場合に true を返します。

それ以外の場合には確認を行い、その結果を返します。:

function checkAge(age) {
  if (age > 18) {
    return true;
  } else {
    return confirm('Do you have your parents permission to access this page?');
  }
}

それを書き直し、1行で if なしで同じをことを実行してください。

checkAge の2つのバリアントを作ってください。:

  1. 疑問符演算子 '?' を使うケース
  2. OR || を使うケース

疑問符演算子 '?' を利用:

function checkAge(age) {
  return (age > 18) ? true : confirm('Did parents allow you?');
}

OR || を利用(最も短いバリアント):

function checkAge(age) {
  return (age > 18) || confirm('Did parents allow you?');
}

age > 18 の周りの括弧はここでは必須ではないことに留意してください。より良い可読性のために存在しています。

重要性: 1

2つの数値 ab で小さい方を返す関数 min(a,b) を書きなさい。

例:

min(2, 5) == 2
min(3, -1) == -1
min(1, 1) == 1

if を使った方法:

function min(a, b) {
  if (a < b) {
    return a;
  } else {
    return b;
  }
}

疑問符演算子 '?' を使った方法:

function min(a, b) {
  return a < b ? a : b;
}

P.S. 等しい a == b 場合、何を返すかは気にする必要ありません。

重要性: 4

x の累乗 n を返す関数 pow(x,n) を書いてください。 つまり、 x をそれ自身で n 回掛け、その結果を返します。

pow(3, 2) = 3 * 3 = 9
pow(3, 3) = 3 * 3 * 3 = 27
pow(1, 100) = 1 * 1 * ...*1 = 1

xn を聞くプロンプトをもつ web ページを作り、pow(x,n) の結果を表示しなさい。

デモを実行

P.S. このタスクでは、n に自然数のみをサポートします。

function pow(x, n) {
  let result = x;

  for (let i = 1; i < n; i++) {
    result *= x;
  }

  return result;
}

let x = prompt("x?", '');
let n = prompt("n?", '');

if (n <= 1) {
  alert(`Power ${n} is not supported,
    use an integer greater than 0`);
} else {
  alert( pow(x, n) );
}
チュートリアルマップ

コメント

コメントをする前に読んでください…
  • 自由に記事への追加や質問を投稿をしたり、それらに回答してください。
  • 数語のコードを挿入するには、<code> タグを使ってください。複数行の場合は <pre> を、10行を超える場合にはサンドボックスを使ってください(plnkr, JSBin, codepen…)。
  • 記事の中で理解できないことがあれば、詳しく説明してください。