2009/09/16
カテゴリ : Tech
JavaScript
配列
PHP
ECMAScript
JavaScriptの連想配列(ハッシュ)はArrayオブジェクトでも作成できる!?
こんにちは、橋本です。
今日は、JavaScriptの連想配列(ハッシュ)に関するお話をしたいと思います。
基本的にJavaScriptで連想配列を作るときには、Objectオブジェクトを使用します。
こんな感じ。
キーと値を取るときはこんな感じ。
基本的に配列のキーに文字列を設定することは出来ません。
…と思い込み、今まではこのやり方に特に疑問を感じずにやってきました。
ところが、今日とあるコードにArrayオブジェクトで連想配列を定義している記述が。
「そ、そんな馬鹿な!!ちゃんと動いているはずが…」と思ったのですが、問題なく動いているようで。
ただ、配列のキーに文字列を指定した場合、lengthプロパティには反映されず、alertで配列の中身を表示することも出来ない模様。
これはどういうことなのだろうと思い、いろいろ試してみました。
どうやら配列のキーに数字を指定した場合のみ、配列の値として認識されるようです。
これはなぜか。
結論から言うと、配列のキーに文字列を指定した場合には、配列の要素としてではなく、Arrayオブジェクトのプロパティとして値が定義されます。
つまり、
とやるのは、
とやるのと同じということです。
Objectオブジェクトで連想配列を定義した場合と同様に、プロパティとして定義されるんですね。
ちなみに、ECMAScriptの仕様書を読むと、lengthプロパティのカウントには、内部メソッドのputメソッドを使っているようです。
(参考: Under Translation of ECMA-262 3rd Edition 15.4 Array オブジェクト (Array Objects))
次のように書かれています。
[[Put]] (P, V)
Array オブジェクトは変化した [[Put]] メソッドを用いて他の Native ECMAScript オブジェクトのために使用される。
A を Array オブジェクト、 P を文字列と想定する。
A の [[Put]] メソッドが、プロパティ P と値 V で呼出されるとき、次のステップが取られる:
1. A の [[CanPut]] method を名前 P で呼出す。
2. Result(1) が false ならば、戻る。
3. A 名前 P のプロパティを持たないならば、ステップ 7 へ。
4. P が "length" ならば、ステップ 12 へ。
5. A のプロパティ P を V に設定する。
6. ステップ 8 ヘ。
7. 名前 P のプロパティを生成し、値を V に設定し空の属性を与える。
8. P が配列の添え字でなければ、戻る。
9. ToUint32(P) が A の length プロパティ未満ならば、戻る。
10. A の length プロパティを ToUint32(P)+1 に変更 (または設定) する。
11. 戻る。
12. ToUint32(V) を算出する。
13. Result(12) が ToNumber(V) と等しくなければ、例外 RangeError を投げる。
14. A の length プロパティ の値未満であり Result(12) 未満でない各整数 k について、 A 自身が ToString(k) という名前の (継承したプロパティではない) プロパティを持つならば、そのプロパティを削除する。
15. A のプロパティ P の値を Result(12) に設定する。
16. 戻る。
ざっくり言うと、キーが数字だったらlength変更するけど、そうじゃなかったら、プロパティとして登録するだけって感じみたいですね。
普段PHPを使っていると、JavaScriptの配列でも連想配列を使用することが出来ると思い込みがち。
Arrayオブジェクトを連想配列として使用することも出来ますが、上記のとおり、lengthプロパティに反映されないなど、いろいろ弊害があるかと思います。
ちなみに、さきほど記載した「キー数字&文字列」で作成した配列をfor~in文とlengthプロパティを使用したfor文で表示すると、それぞれ違った結果になります。
紛らわしいので、やっぱり連想配列を使うときには、Objectオブジェクトを使った方がよさげですね。
今日は、JavaScriptの連想配列(ハッシュ)に関するお話をしたいと思います。
基本的にJavaScriptで連想配列を作るときには、Objectオブジェクトを使用します。
こんな感じ。
- var
hoge = { hoge: 'hoge', fuga: 'fuga' - };
キーと値を取るときはこんな感じ。
- for
(var a in hoge) { //キー alert(a); // 値 alert(hoge[a]); - }
基本的に配列のキーに文字列を設定することは出来ません。
…と思い込み、今まではこのやり方に特に疑問を感じずにやってきました。
ところが、今日とあるコードにArrayオブジェクトで連想配列を定義している記述が。
「そ、そんな馬鹿な!!ちゃんと動いているはずが…」と思ったのですが、問題なく動いているようで。
ただ、配列のキーに文字列を指定した場合、lengthプロパティには反映されず、alertで配列の中身を表示することも出来ない模様。
これはどういうことなのだろうと思い、いろいろ試してみました。
- //1.
キー数字のみ - var
hoge = new Array(); - hoge[0]
= 'hoge'; - hoge[1]
= 'fuga'; - alert(hoge.length);
// 2
- //2.
キー文字列のみ - var
hoge = new Array(); - hoge['hoge']
= 'hoge'; - hoge['fuga']
= 'fuga'; - alert(hoge.length);
// 0
- //3.
キー数字&文字列 - var
hoge = new Array(); - hoge['hoge']
= 'hoge'; - hoge['fuga']
= 'fuga'; - hoge[0]
= 'puyo'; - hoge[1]
= 'aaaa'; - alert(hoge.length);
// 2
どうやら配列のキーに数字を指定した場合のみ、配列の値として認識されるようです。
これはなぜか。
結論から言うと、配列のキーに文字列を指定した場合には、配列の要素としてではなく、Arrayオブジェクトのプロパティとして値が定義されます。
つまり、
- var
hoge = new Array(); - hoge['fuga']
= 'hoge';
とやるのは、
- var
hoge = new Array(); - hoge.fuga
= 'hoge';
とやるのと同じということです。
Objectオブジェクトで連想配列を定義した場合と同様に、プロパティとして定義されるんですね。
ちなみに、ECMAScriptの仕様書を読むと、lengthプロパティのカウントには、内部メソッドのputメソッドを使っているようです。
(参考: Under Translation of ECMA-262 3rd Edition 15.4 Array オブジェクト (Array Objects))
次のように書かれています。
[[Put]] (P, V)
Array オブジェクトは変化した [[Put]] メソッドを用いて他の Native ECMAScript オブジェクトのために使用される。
A を Array オブジェクト、 P を文字列と想定する。
A の [[Put]] メソッドが、プロパティ P と値 V で呼出されるとき、次のステップが取られる:
1. A の [[CanPut]] method を名前 P で呼出す。
2. Result(1) が false ならば、戻る。
3. A 名前 P のプロパティを持たないならば、ステップ 7 へ。
4. P が "length" ならば、ステップ 12 へ。
5. A のプロパティ P を V に設定する。
6. ステップ 8 ヘ。
7. 名前 P のプロパティを生成し、値を V に設定し空の属性を与える。
8. P が配列の添え字でなければ、戻る。
9. ToUint32(P) が A の length プロパティ未満ならば、戻る。
10. A の length プロパティを ToUint32(P)+1 に変更 (または設定) する。
11. 戻る。
12. ToUint32(V) を算出する。
13. Result(12) が ToNumber(V) と等しくなければ、例外 RangeError を投げる。
14. A の length プロパティ の値未満であり Result(12) 未満でない各整数 k について、 A 自身が ToString(k) という名前の (継承したプロパティではない) プロパティを持つならば、そのプロパティを削除する。
15. A のプロパティ P の値を Result(12) に設定する。
16. 戻る。
ざっくり言うと、キーが数字だったらlength変更するけど、そうじゃなかったら、プロパティとして登録するだけって感じみたいですね。
普段PHPを使っていると、JavaScriptの配列でも連想配列を使用することが出来ると思い込みがち。
Arrayオブジェクトを連想配列として使用することも出来ますが、上記のとおり、lengthプロパティに反映されないなど、いろいろ弊害があるかと思います。
ちなみに、さきほど記載した「キー数字&文字列」で作成した配列をfor~in文とlengthプロパティを使用したfor文で表示すると、それぞれ違った結果になります。
- var
hoge = new Array(); - hoge['hoge']
= 'hoge'; - hoge['fuga']
= 'fuga'; - hoge[0]
= 'puyo'; - hoge[1]
= 'aaaa'; - //
for~in文 - for
(var i in hoge) { alert(hoge[i]); //'hoge','fuga','puyo','aaaa'が表示される - }
- //
lengthプロパティを使用したfor文 - for
(var i = 0; i < hoge.length; i++) { alert(hoge[i]); //'puyo','aaaa'が表示される - }
紛らわしいので、やっぱり連想配列を使うときには、Objectオブジェクトを使った方がよさげですね。
トラックバックURI
-
- JavaScript Object from JustOnePlanet Blog
- ■オブジェクトを生成 オブジェクトリテラルを使った生成 世間的には...
2009年09月19日 05:43
最近の記事
システム開発エンジニア募集! [2012年02月03日 : 小林有佳]
OpenVPNで細々便利な設定 [2012年01月31日 : 門脇優児]
【iOS】Viewの開発・デバッグに役立つ色々 [2012年01月23日 : 中川善樹]
PHPDocumentorの利用方法まとめ [2012年01月19日 : 笹亀弘]
Google Chart Toolsを使ってサイトマップを描こう! [2011年12月21日 : 志田仁美]
stumpwm設定v2 [2011年12月19日 : 門脇優児]
Mashup Awards 7の授賞式が行われました [2011年12月16日 : 中川善樹]
社員旅行に行きました [2011年12月12日 : 大橋寛子]
iCloud風のアイコンを作成する(Fireworks) [2011年12月07日 : 和田記光]
iScroll4でネイティブに近いスマホ向けHTMLページを作成する [2011年12月02日 : 松田惇]













コメントフォーム