質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
Google Apps Script

Google Apps ScriptはGoogleの製品と第三者のサービスでタスクを自動化するためのJavaScriptのクラウドのスクリプト言語です。

JavaScript

JavaScriptは、プログラミング言語のひとつです。ネットスケープコミュニケーションズで開発されました。 開発当初はLiveScriptと呼ばれていましたが、業務提携していたサン・マイクロシステムズが開発したJavaが脚光を浴びていたことから、JavaScriptと改名されました。 動きのあるWebページを作ることを目的に開発されたもので、主要なWebブラウザのほとんどに搭載されています。

Q&A

解決済

1回答

521閲覧

GASで動的にオブジェクトを作成し、値を格納する

kashim

総合スコア38

Google Apps Script

Google Apps ScriptはGoogleの製品と第三者のサービスでタスクを自動化するためのJavaScriptのクラウドのスクリプト言語です。

JavaScript

JavaScriptは、プログラミング言語のひとつです。ネットスケープコミュニケーションズで開発されました。 開発当初はLiveScriptと呼ばれていましたが、業務提携していたサン・マイクロシステムズが開発したJavaが脚光を浴びていたことから、JavaScriptと改名されました。 動きのあるWebページを作ることを目的に開発されたもので、主要なWebブラウザのほとんどに搭載されています。

0グッド

0クリップ

投稿2022/08/13 02:20

前提

GAS

1 const obj = {}; 2 const data = new Map([ 3 ['event.AAA.B','テスト1'], 4 ['event','テスト2'], 5 ['id','テスト3'], 6 ['data.3.B.a','テスト4'] 7 ]);

というデータがあったときに、

GAS

1 obj['event']['AAA']['B'] = 'テスト1'; 2 obj['event'] = 'テスト2'; 3 obj['id'] = 'テスト3'; 4 obj['data']['3']['B']['a'] = 'テスト4'; 5 6// Logger.log(obj); 7// {event=テスト2, id=テスト3}

となるようにデータ格納したいです。

実現したいこと

1.配列内のkeyを.で分割し、それをobjの配列のkeyにしたいです
(key側についている.の前後の値も数も変化します)
2..がない場合、objにそのまま格納

該当のソースコード

GAS

1function myFunction() { 2 const obj = {}; 3 const data = new Map([ 4 ['event.AAA.B','テスト1'], 5 ['event','テスト2'], 6 ['id','テスト3'], 7 ['data.3.B.a','テスト4'] 8 ]); 9 10 data.forEach((item, index) => { 11 var arr = index.split("."); 12 13 // 分割数を確認 14 var arr_size = arr.length; 15 16 // 1個しかない為、そのまま 17 if(arr_size == 1){ 18 obj[index] = item; 19 }else{ 20 var val_count = 0; 21 22 arr.forEach(function(key){ 23 val_count += 1; 24 // 最後の値 25 if(val_count == arr_size){ 26 // 最後の値が取得出来ている 27 // Logger.log(key); 28 } 29 }); 30 31 // Logger.log(obj); 32 // {event=テスト2, id=テスト3} 33 } 34 }); 35} 36

エラーメッセージ

ただ単にそのまま格納しようとした際に

obj['event']['AAA']['B'] = 'テスト1';

を実行しようとすると当然ですが、eventが無かったりAAAが存在しないということで
TypeError: Cannot read property 'AAA' of undefined
を吐いてしまう為、この辺りをどうやってクリアするかが思いつきませんでした。

試したこと

1.MAP内のkeyの値を.で分割まで完了しました
2..が無い場合、そのままobjに格納するまで完了しました
3.最後のピリオドだった場合の分岐を作成しました

動的にオブジェクトを作成するのが難しく
詳しい方いらっしゃったらよろしくお願いいたします。
アドバイスだけでも本当に助かります。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

obj['event']['AAA']['B'] = 'テスト1'; obj['event'] = 'テスト2';

仮にこれが

console.log( obj['event']['AAA']['B']) を実行した時に 「テスト1」 と表示し、 かつ console.log( obj['event']) を実行した時に 「テスト2」 と表示したい

という意味ならば、このようなこと(同一オブジェクトの同一のキーに対して異なる値を割り当てること)は、Javascriptでは実現できないと思います。

上のような場合を除外して質問文のような動作をさせたい場合は、たとえば下記のようになるかと思います。
(おそらくもっとすっきりした書き方はあると思いますが)

js

1function con(ary) { 2 if (Object.prototype.toString.call(ary) !== '[object Array]') 3 throw TypeError('引数は配列でなければなりません'); 4 if (ary.length < 2) 5 throw TypeError('長さが2未満の配列は指定できません'); 6 if (ary.length === 2) { 7 return Object.fromEntries([[ary[0], ary[1]]]); 8 } 9 return { [ary[0]]: con(ary.slice(1)) } 10} 11 12function myFunction() { 13 let obj = {}; 14 const data = new Map([ 15 ['event.AAA.B', 'テスト1'], 16 // ['event','テスト2'], // キー('event')が1つ前と重複しているためコメントアウト 17 ['id', 'テスト3'], 18 ['data.3.B.a', 'テスト4'] 19 ]); 20 21 for ([v, k] of data.entries()) { 22 obj = { ...obj, ...con([...v.split('.'), k]) }; 23 } 24 console.log(JSON.stringify(obj)); 25}

エラー処理を無視するなら、con関数は下記のようにもできます

js

1const con = ary => ({ [ary[0]]: ary.length === 2 ? ary[1] : con(ary.slice(1)) });

投稿2022/08/13 06:09

編集2022/08/13 06:23
退会済みユーザー

退会済みユーザー

総合スコア0

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

kashim

2022/08/13 06:16

ありがとうございます!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問