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

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

新規登録して質問してみよう
ただいま回答率
85.48%
Vue.js

Vue.jsは、Webアプリケーションのインターフェースを構築するためのオープンソースJavaScriptフレームワークです。

JavaScript

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

Q&A

解決済

1回答

1243閲覧

[JavaScript]ForOfの処理をPromise.allに移行したいです。

yurika_21

総合スコア12

Vue.js

Vue.jsは、Webアプリケーションのインターフェースを構築するためのオープンソースJavaScriptフレームワークです。

JavaScript

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

0グッド

1クリップ

投稿2022/09/10 09:23

前提

Vue2系でアップロードの処理を開発しているのですが、async.awaitを使用するためにForOfで処理するように記述しています。
しかし、並列処理にしたい観点からForOfからPromise.allに移行しようと考えております。

実現したいこと

ForOfで行っている処理をPromise.allに移行して同じ動作がするようにしたいです。

発生している問題・エラーメッセージ

Promise.allで書き換えてみたのですが、上手くアップロード処理が行われず(主にUpdate処理が)、書き方が間違っている気がしてならないです。 エラー等は吐いていません。

該当のソースコード

JavaScript

1async uploadAlbum() { 2 //forEachはasync・await出来ないからforOfで 3 for (const [index, file] of Object.entries(this.imageFiles)) { 4 if (!file) { 5 return; 6 } 7 const fileType = file.type.replace("image/", ""); 8 const currentUser = getAuth().currentUser; 9 const relativePath = `user/${currentUser.uid}/${index}.${fileType}`; 10 this.imageFiles[index] = await this.mixinUploader__upload( 11 file, 12 relativePath 13 ); 14 } 15 // forOfで書いた場合 16 // await Promise.all( 17 // Object.values(this.imageFiles).map(async (file, index) => { 18 // if (!file) { 19 // return; 20 // } 21 // console.error(file); 22 // const count = index + 1; 23 // const fileType = file.type.replace("image/", ""); 24 // const currentUser = getAuth().currentUser; 25 // const relativePath = `user/${currentUser.uid}/${count}.${fileType}`; 26 // this.imageFiles[count] = 27 // await this.mixinUploader__upload(file, relativePath); 28 // }) 29 // ); 30 },

試したこと

Promise.all で自分なりに書いてみたが書き換えが上手く動作しない

補足情報(FW/ツールのバージョンなど)

Vue2系

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

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

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

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

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

maisumakun

2022/09/10 09:27

> 上手くアップロード処理が行われず 何が行われて何が行われていない状況なのか、もう少し詳細をいただくことはできますでしょうか。
guest

回答1

0

ベストアンサー

修正の要点としては下記です。

  1. Promise.all() には引数としてPromiseの配列を渡す必要がある。
  2. 1.のPromise の配列を map で作るのであれば、map に渡す関数はどの場合でもProimseまたは (if(!file) の場合には)即resolveされる値を返す必要がある。
  3. 1.のPromiseの配列の要素となるのは this.mixinUploader__upload(file, relativePath) の返すPromiseが元になる。ただし次の 4. に注意する。
  4. アップロード結果から、this.imageFiles を再構成できるようにするために、 this.mixinUploader__upload が返すPromiseがresolveされて得られる値(つまりアップロード処理の成功時のレスポンス)を、 this.imageFilesオブジェクトのキーである index と紐付けた、キーと値のペアが最終的にresolveされる値として必要
  5. 4.によってindexとアップロード結果のペアの配列が得られたら、これを元にObject.fromEntries() して得られるオブジェクトでthis.imageFilesをアップロード結果に更新できる。

上記を踏まえるとuploadAlbum()メソッドは以下のような感じになるかと思います。

javascript

1 async uploadAlbum() { 2 const uploadingPromises = 3 Object.entries(this.imageFiles).map(([index, file]) => { 4 if (!file) { 5 return [index, { error: new Error('Oops! file is empty 😓') }]; 6 } 7 const fileType = file.type.replace("image/", ""); 8 const currentUser = getAuth().currentUser; 9 const relativePath = `user/${currentUser.uid}/${index}.${fileType}`; 10 return this.mixinUploader__upload( 11 file, 12 relativePath 13 ).then(uploadResult => [index, uploadResult]); 14 }); 15 try { 16 const fileEntriesAfterUpload = await Promise.all(uploadingPromises); 17 this.imageFiles = Object.fromEntries(fileEntriesAfterUpload); 18 } catch (e) { 19 console.error(e); 20 } 21 }

投稿2022/09/10 11:27

編集2022/09/10 11:32
退会済みユーザー

退会済みユーザー

総合スコア0

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問