質問するログイン新規登録

回答編集履歴

7

テキスト修正

2019/09/16 17:02

投稿

jun68ykt
jun68ykt

スコア9058

answer CHANGED
@@ -109,7 +109,7 @@
109
109
 
110
110
  - MDN: オブジェクト初期化子 - [ECMAScript 2015 での新しい表記法](https://developer.mozilla.org/ja/docs/Web/JavaScript/Reference/Operators/Object_initializer#New_notations_in_ECMAScript_2015)
111
111
 
112
- 上記は、以下の3つ
112
+ 上記のMDNの説明には、以下の3つ
113
113
 
114
114
  - Shorthand property names
115
115
  - Shorthand method names

6

テキスト修正

2019/09/16 17:02

投稿

jun68ykt
jun68ykt

スコア9058

answer CHANGED
@@ -197,7 +197,7 @@
197
197
   ・・・
198
198
  }
199
199
  ```
200
- と宣言された関数 `init0` を実行することを考えます。最も簡単には、`init0`を実行するには
200
+ と宣言された関数 `init0` を実行することを考えます。最も簡単に関数`init0`を実行するには
201
201
  ```javascript
202
202
  init0();
203
203
  ```

5

テキスト修正

2019/09/16 15:33

投稿

jun68ykt
jun68ykt

スコア9058

answer CHANGED
@@ -261,7 +261,7 @@
261
261
 
262
262
  手前味噌で恐縮ですが、この本に掲載されているサンプルコードの一覧を以下に作成しています。
263
263
 
264
- - Qiita: [「開眼!JavaScript」サンプルコード#1/4 第1-4章](https://qiita.com/jun68ykt/items/5aee04682f95f2354398) (※5章以降は、のページからリンクされた別記事です。)
264
+ - Qiita: [「開眼!JavaScript」サンプルコード#1/4 第1-4章](https://qiita.com/jun68ykt/items/5aee04682f95f2354398) (※5章以降は、左記のページからリンクされた別記事です。)
265
265
 
266
266
  上記の記事には、この本のサンプルコードへのリンクを一覧しているので、各節のタイトルとコードを見て頂けると、どんな本か、ざっくりつかめると思います。ちなみに
267
267
 

4

テキスト修正

2019/09/16 14:25

投稿

jun68ykt
jun68ykt

スコア9058

answer CHANGED
@@ -41,4 +41,286 @@
41
41
  window['init0']();
42
42
  ```
43
43
  でもよいです。
44
- - **動作確認用CodePen:** [https://codepen.io/jun68ykt/pen/eYOjgjW?editors=0012](https://codepen.io/jun68ykt/pen/eYOjgjW?editors=0012)
44
+ - **動作確認用CodePen:** [https://codepen.io/jun68ykt/pen/eYOjgjW?editors=0012](https://codepen.io/jun68ykt/pen/eYOjgjW?editors=0012)
45
+
46
+
47
+ ### 追記2
48
+
49
+ 質問者さまからのコメントを、以下
50
+
51
+ (1) コードについての説明
52
+
53
+ > 今回教えていただいた方法について仕組みが今ひとつ理解できません…。
54
+
55
+
56
+ (2) 参考資料、図書について
57
+
58
+ > 今回教えていただいた手法をちゃんと理解するのに役立つ参考サイト・文献など
59
+
60
+ の2点に分けて、それぞれに回答します。
61
+
62
+ #### (1) コードについての説明
63
+
64
+ 以下について回答します。
65
+
66
+ > 教えていただいたものだとinit0,init1,init2 という文字の配列のように見えます。。。
67
+ > また、「initializers[str]();」これで関数を実行できるというやり方も今回初めて目にしたのですが
68
+ こういう方法は何か名前がついていますか? 
69
+
70
+
71
+ まず、
72
+
73
+ > 教えていただいたものだとinit0,init1,init2 という文字の配列のように見えます。。。
74
+
75
+ についてです。
76
+ `top.js` に、関数名が `init` で始まる、以下の3つの関数が宣言されています。
77
+
78
+ ```javascript
79
+ function init0() {
80
+  ・・・
81
+ }
82
+
83
+ function init1() {
84
+  ・・・
85
+ }
86
+
87
+ function init2() {
88
+  ・・・
89
+ }
90
+ ```
91
+ これらに対して、先の回答に書いた、`var initializers` に代入している以下のオブジェクトリテラル
92
+ ```javascript
93
+ {
94
+ init0,
95
+ init1,
96
+ init2
97
+ }
98
+ ```
99
+ は、以下
100
+ ```javascript
101
+ {
102
+ init0: init0,
103
+ init1: init1,
104
+ init2: init2
105
+ }
106
+ ```
107
+ を省略した書き方です。このような省略した書き方ができるようになったのは ECMAScript 2015 からです。
108
+ MDN では、以下に説明があります。
109
+
110
+ - MDN: オブジェクト初期化子 - [ECMAScript 2015 での新しい表記法](https://developer.mozilla.org/ja/docs/Web/JavaScript/Reference/Operators/Object_initializer#New_notations_in_ECMAScript_2015)
111
+
112
+ 上記では、以下の3つ
113
+
114
+ - Shorthand property names
115
+ - Shorthand method names
116
+ - Computed property names
117
+
118
+ のコード例が挙げられていますが、これらのうち
119
+ ```javascript
120
+ {
121
+ init0,
122
+ init1,
123
+ init2
124
+ }
125
+ ```
126
+ は、1番目の `Shorthand property names` を使った書き方です。
127
+
128
+ 次に
129
+
130
+ > また、「initializers[str]();」これで関数を実行できるというやり方も今回初めて目にしたのですが
131
+ こういう方法は何か名前がついていますか? 
132
+
133
+ についてですが、ちょっと端折った書き方をしてしまったので、何か特別なことをやってるように見えてしまったかもしれませんが、`initializers[str]();` をもう少し丁寧に書くと、以下になります。
134
+
135
+ ```javascript
136
+ const fn = initializers[str];
137
+
138
+ fn();
139
+ ```
140
+ 上記では
141
+ (ⅰ) 変数 `fn` に、 `initializers[str]` を代入する。
142
+ (ⅱ) 上記の代入によって、変数 `fn` には関数が入ってくる想定なので、`fn()` で、この関数を実行する。
143
+ という2段階のことを行っており、それぞれに何か特別なことをやっているわけではないので
144
+
145
+ > 何か名前
146
+
147
+ というものがついているわけではなく、上記で、いったん `fn` に入れているのを、端折って書けば、
148
+ ```javascript
149
+ initializers[str]();
150
+ ```
151
+ というものになる、ということです。
152
+
153
+ 上記 (ⅰ)、(ⅱ) に関連する JavaScript の基礎知識あるいは概念としては、
154
+
155
+ (ⅰ) プロパティアクセサー
156
+
157
+ (ⅱ) javascript の関数は、第一級オブジェクト
158
+
159
+ が挙げられます。以下、各々について説明します。
160
+
161
+
162
+ **(ⅰ) プロパティアクセサー**
163
+
164
+ MDN のプロパティアクセサーの説明は以下です。
165
+
166
+ - MDN: [プロパティアクセサー](https://developer.mozilla.org/ja/docs/Web/JavaScript/Reference/Operators/Property_Accessors)
167
+
168
+
169
+ 上記に以下のように書かれています。
170
+
171
+ > プロパティアクセサーはオブジェクトのプロパティへのアクセスを提供するもので、ドット表記法またはブラケット表記法を使用します。
172
+
173
+ 今回のご質問では、 `"init0"` または `"init1"`, `"init2"` という文字列が入っている想定の変数 `str` があったとして、`initializers` のプロパティの中で、`str`の値と一致するプロパティの値が欲しいわけですが、その際にはドット表記法で `initializers.str` **とは書けず、 ** ブラケット表記法で、
174
+ ```javascript
175
+ initializers[str]
176
+ ```
177
+ と書きます。
178
+
179
+ **(ⅱ) JavaScript の関数は、第一級オブジェクト**
180
+
181
+ MDN の[関数の説明](https://developer.mozilla.org/ja/docs/Web/JavaScript/Reference/Functions) の冒頭、2段落目に以下のように書かれています。
182
+
183
+ > JavaScript において、関数は第一級オブジェクトです。すなわち、関数はオブジェクトであり、他のあらゆるオブジェクトと同じように操作したり渡したりする事ができます。
184
+
185
+ また、 wikipedia の [第一級オブジェクトの説明](https://ja.wikipedia.org/wiki/%E7%AC%AC%E4%B8%80%E7%B4%9A%E3%82%AA%E3%83%96%E3%82%B8%E3%82%A7%E3%82%AF%E3%83%88) に、以下の記載があります。
186
+
187
+ > 第一級オブジェクトは概ね次のような性質をもつ。
188
+ > ・・・
189
+ > ・変数に格納可能である
190
+ > ・データ構造に格納可能である。
191
+ > ・・・
192
+
193
+ JavaScript の関数が上記の性質を持つことを踏まえて、
194
+
195
+ ```javascript
196
+ function init0() {
197
+  ・・・
198
+ }
199
+ ```
200
+ と宣言された関数 `init0` を実行することを考えます。最も簡単には、`init0`を実行するには
201
+ ```javascript
202
+ init0();
203
+ ```
204
+ と書けばよいのですが、その他にも、以下のように `init0` を 変数`f` に代入して、 `f(); `と書くことでも実行できます。
205
+ ```javascript
206
+ const f = init0;
207
+
208
+ f();
209
+ ```
210
+
211
+ あるいは、`init0` を、何らかのオブジェクトのプロパティの値にして、そのオブジェクトのメソッドとして実行することもできます。
212
+
213
+ ```javascript
214
+ const obj = { f: init0 };
215
+
216
+ obj.f();
217
+ ```
218
+
219
+ このように、 `init0` を他の変数に入れたり、何らかのオブジェクトのプロパティの値にしたりできますし、また、何らか別の関数が return する値の中で、 init0 を使うこともできます。たとえば以下の関数
220
+
221
+ ```javascript
222
+ function getInitializers() {
223
+ return [ init0, init1, init2 ];
224
+ }
225
+ ```
226
+
227
+ は、 長さが3で、要素が `init0`, `init1`, `init2` であるような配列を返します。この関数`getInitializers()` を呼ぶ側では、
228
+
229
+ ```javascript
230
+ const initFuncs = getInitializers();
231
+
232
+ initFuncs[1](); // init1() が実行される。
233
+ ```
234
+
235
+ といったように使います。
236
+
237
+ 関数が第一級オブジェクトであるので、分かり易くいうと、
238
+
239
+ **関数を変数やプロパティや配列の要素に入れて、どこにでも持ち運べる。**
240
+
241
+ といった感じです。
242
+
243
+
244
+ #### (2) 参考資料、図書について
245
+
246
+
247
+ 先述した **(1) コードについての説明** を整理すると、以下の3つのトピックがありました。
248
+
249
+ **(A) Shorthand property names (ECMAScript 2015)**
250
+ **(B) プロパティアクセサー**
251
+ **(C) 関数が第一級オブジェクトであること**
252
+
253
+ 上記3点についての参考資料、推薦図書を以下に挙げます。
254
+
255
+ (A) の参考資料は後述します。
256
+ (B) プロパティアクセサー と、(C) 関数が第一級オブジェクトであることについては、ECMAScript 2015よりも前からJavaScript に備わっている、基礎事項に含まれる項目ですが、これらについてきちんと理解をするための参考書として、個人的に推薦するのは、以下です。
257
+
258
+ - [「開眼! JavaScript ―言語仕様から学ぶJavaScriptの本質」オライリージャパン](https://www.amazon.co.jp/dp/487311621X/)
259
+
260
+ 基礎の何らかのトピックを習得しようとするときに、自分でそのトピックだけにフォーカスした(短い)コードを書いてその結果を確認し、さらにコードを修正して結果の違いを理解するという作業が欠かせませんが、白紙から自分でその理解のためのコードを書くのは大変です。この本は、そのようなコードを多数、サンプルで提供してくれる点がお勧めの理由です。また、内容の濃さに比べて薄い本なのも、取り組みやすいです。
261
+
262
+ 手前味噌で恐縮ですが、この本に掲載されているサンプルコードの一覧を以下に作成しています。
263
+
264
+ - Qiita: [「開眼!JavaScript」サンプルコード#1/4 第1-4章](https://qiita.com/jun68ykt/items/5aee04682f95f2354398) (※5章以降は、先のページからリンクされた別記事です。)
265
+
266
+ 上記の記事には、この本のサンプルコードへのリンクを一覧しているので、各節のタイトルとコードを見て頂けると、どんな本か、ざっくりつかめると思います。ちなみに
267
+
268
+ **(B) プロパティアクセサー**
269
+
270
+ については、この本では
271
+
272
+ - 2.3節 ドット記法とブラケット記法でオブジェクトのプロパティにアクセス
273
+
274
+ に説明とサンプルコードがあります。また、
275
+
276
+ **(C) 関数が第一級オブジェクトであること**
277
+
278
+ については、
279
+
280
+ - 4.6節 JavaScriptの関数は第一級関数
281
+
282
+ に説明とサンプルコードが載っています。
283
+
284
+ ただ、この本について一つ、難点があります。上記の Qiita の記事に一覧したリンクのどれかをクリックして頂くと分かりますが、サンプルコードは jsfiddle に作成されており、 console.log の出力を見ることでコードの動作を確認することになっており、発刊当初はもちろん console.log で動作確認できていたのですが、今現在では console に出力するための firebug-lite-debug が動かないので、コードを自分の手元の開発環境にコピペするか、javascriptのコードを動かせる他のサービス、例えばCodePenなどにコピペするかしないと console.logへの出力で動作の確認ができなくなっています。そこは一手間かかってしまいますが、その手間を差し引いても、重要な概念について、サンプルコードを多数提供してくれるこの本は良書と思います。
285
+
286
+ 次に
287
+
288
+ **(A) Shorthand property names (ECMAScript 2015)**
289
+
290
+ についての参考図書を挙げるとすると、ECMAScript 2015 以降に新たに導入された書き方や機能に集中した内容となっている書籍として、以下をお勧めします。
291
+
292
+ - [「入門 JavaScript プログラミング」翔泳社](https://www.amazon.co.jp/%E5%85%A5%E9%96%80JavaScript%E3%83%97%E3%83%AD%E3%82%B0%E3%83%A9%E3%83%9F%E3%83%B3%E3%82%B0-JD-Isaacks/dp/479815864X)
293
+
294
+ この本は、 上記のとおり邦題が「入門 JavaScript プログラミング」ですが、これは原著のタイトルである
295
+
296
+ Get Programming with JavaScript **Next**
297
+
298
+ の末尾にある "Next" を訳しきれていないので、まったくの入門書に誤解されてしまいますが、原著の副題は
299
+
300
+ New feature of ECMAScript 2015, 2016 and beyond
301
+
302
+ です。
303
+ また、和訳版まえがきの「本書について」には、こう書かれています。
304
+
305
+ > 本書は、ES2015 以降に導入された新しい機能をこれから学ぼうとしているJavaScript プログラマのために書かれている。
306
+
307
+ また、「本書の対象読者」には以下のようにも書かれています。
308
+
309
+ > 従来のJavaScriptを使って問題なくプログラミングを行えることが前提となるが、・・・
310
+
311
+ 上記にある、「従来のJavaScriptを使って問題なくプログラミングを行えること」 の補強のために、先の「開眼! JavaScript ―言語仕様から学ぶJavaScriptの本質」をお勧めする次第です。
312
+
313
+ ただ、この本で、ECMAScript 2015 以降の書き方のすべてが網羅されているかというと、そのような総覧ではないので、一覧としては個人的には以下を使うことがあります。
314
+
315
+ - [ECMAScript 6 — New Features: Overview & Comparison](http://es6-features.org)
316
+
317
+
318
+ 上記のサイトでは、先に挙げた
319
+ (A) Shorthand property names (ECMAScript 2015)
320
+ は、 以下に説明とコード例があります。
321
+
322
+ - [http://es6-features.org/#PropertyShorthand](http://es6-features.org/#PropertyShorthand)
323
+
324
+ 上記のサイトでは、「Ecmascript 5 以前ではこう書いていたけれど Ecmascript 2015(ES6)ではこのように書けるようになった」という、新旧のコード対比ができるので分かり易いです。
325
+
326
+ 以上、参考になれば幸いです。

3

テキスト修正

2019/09/16 14:18

投稿

jun68ykt
jun68ykt

スコア9058

answer CHANGED
@@ -33,9 +33,9 @@
33
33
  ```
34
34
  - **動作確認用CodePen:** [https://codepen.io/jun68ykt/pen/KKPBKbN?editors=0012](https://codepen.io/jun68ykt/pen/KKPBKbN?editors=0012)
35
35
 
36
- または、WEBブラウザで動かしているコードのでしたら、グローバルオブジェクトを得るのに、
36
+ または、ご質問のコードは、WEBブラウザで動かコードですので、グローバルオブジェクトを得るのに、
37
37
  `var global = (function(){ return this; })();`
38
- とまわりくどいことをせずに、 グローバルオブジェクトが `window` であること分かっているので、
38
+ とまわりくどいことをせずに、 グローバルオブジェクトが `window` であること分かっているので、
39
39
 
40
40
  ```javascript
41
41
  window['init0']();

2

テキスト修正

2019/09/15 02:27

投稿

jun68ykt
jun68ykt

スコア9058

answer CHANGED
@@ -31,4 +31,14 @@
31
31
 
32
32
  global[str]();
33
33
  ```
34
- - **動作確認用CodePen:** [https://codepen.io/jun68ykt/pen/KKPBKbN?editors=0012](https://codepen.io/jun68ykt/pen/KKPBKbN?editors=0012)
34
+ - **動作確認用CodePen:** [https://codepen.io/jun68ykt/pen/KKPBKbN?editors=0012](https://codepen.io/jun68ykt/pen/KKPBKbN?editors=0012)
35
+
36
+ または、WEBブラウザで動かしているコードなのでしたら、グローバルオブジェクトを得るのに、
37
+ `var global = (function(){ return this; })();`
38
+ とまわりくどいことをせずに、 グローバルオブジェクトが `window` であることが分かっているので、
39
+
40
+ ```javascript
41
+ window['init0']();
42
+ ```
43
+ でもよいです。
44
+ - **動作確認用CodePen:** [https://codepen.io/jun68ykt/pen/eYOjgjW?editors=0012](https://codepen.io/jun68ykt/pen/eYOjgjW?editors=0012)

1

テキスト修正

2019/09/15 02:20

投稿

jun68ykt
jun68ykt

スコア9058

answer CHANGED
@@ -18,4 +18,17 @@
18
18
 
19
19
  - **動作確認用CodePen:** [https://codepen.io/jun68ykt/pen/GRKBRBa?editors=0012](https://codepen.io/jun68ykt/pen/GRKBRBa?editors=0012)
20
20
 
21
- 以上参考になれば幸いです。
21
+ 以上参考になれば幸いです。
22
+
23
+ ---
24
+ ### 追記
25
+
26
+ (あまり筋の良い方法とは思いませんが、)以下のようにグローバルオブジェクトのプロパティとして各関数を取り出して、これを実行する、ということも可能です。
27
+
28
+ ```
29
+ var global = (function(){ return this; })();
30
+ var str = 'init0';
31
+
32
+ global[str]();
33
+ ```
34
+ - **動作確認用CodePen:** [https://codepen.io/jun68ykt/pen/KKPBKbN?editors=0012](https://codepen.io/jun68ykt/pen/KKPBKbN?editors=0012)