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

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

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

Node.jsとはGoogleのV8 JavaScriptエンジンを使用しているサーバーサイドのイベント駆動型プログラムです。

データベース

データベースとは、データの集合体を指します。また、そのデータの集合体の共用を可能にするシステムの意味を含めます

ループ

ループとは、プログラミングにおいて、条件に合致している間、複数回繰り返し実行される箇所や、その制御構造を指します

非同期処理

非同期処理とは一部のコードを別々のスレッドで実行させる手法です。アプリケーションのパフォーマンスを向上させる目的でこの手法を用います。

Q&A

解決済

1回答

665閲覧

Node:非同期処理の中で、大量の要素すべてに対してそれぞれ処理を行いたい。飛び飛びに処理してしまう。

shimon_

総合スコア26

Node.js

Node.jsとはGoogleのV8 JavaScriptエンジンを使用しているサーバーサイドのイベント駆動型プログラムです。

データベース

データベースとは、データの集合体を指します。また、そのデータの集合体の共用を可能にするシステムの意味を含めます

ループ

ループとは、プログラミングにおいて、条件に合致している間、複数回繰り返し実行される箇所や、その制御構造を指します

非同期処理

非同期処理とは一部のコードを別々のスレッドで実行させる手法です。アプリケーションのパフォーマンスを向上させる目的でこの手法を用います。

0グッド

0クリップ

投稿2020/09/01 15:01

前提

【DB Models】
day_info_today: 指定日の情報が格納されているテーブル。実装済み。
day_info_diff: 指定日とその前日の情報との差分が格納されているテーブル。未実装。

実現したいこと

day_info_todayから指定日とその前日の情報を取得し、差分を計算。その後、day_info_diffに計算結果を格納したい。

試したこと

試してみた処理は以下のようなものです
①day_info_todayから全情報を取得(配列化)
②配列からインデックスで当日と前日の情報を取得し処理
③処理結果をday_info_diffへ格納
④インデックスを+1し次の情報へ→①へ戻る

生じた問題

forでインデックスを回していたのですが、非同期処理であるため、処理が終了する前にforがループしてしまいました。結果、全要素を処理することができませんでした。

質問

実現目標を達成するためには、どのような処理が必要でしょうか?
また、もっとスマートな方法をご存じであるならば、教えていただければ助かります。
よろしくお願いいたします。

DB1を更新したのち全情報を取得、それらを処理するコード

node

1'use strict'; 2/** TotalとDiff、2つのDBに初期値をセットするプログラムです */ 3const fs = require('fs'); 4const csvParse = require('csv-parse'); 5 6// モデルの読み込み 7const Day_info_total = require('../models/day_info_total'); 8const Day_info_diff = require('../models/day_info_diff'); 9Day_info_total.sync(); 10Day_info_diff.sync(); 11 12// DBに格納する情報の大本。要素数約5000個 13const path = './downloads/all/total/all.tsv'; 14 15function setTotalDatas(path) { 16 return new Promise((resolve) => { 17  // tsvをJsonに変換する処理 18 var rs = fs.createReadStream(path, 'utf-8'); 19 var parser = csvParse({ 20 delimiter: '\t', 21 columns: ['date', 'time', 'unixTime', 'videos', 'views', 'comments'], 22 }); 23 rs.pipe(parser).on('data', (data) => { 24 // Day_info_totalを更新する 25 Day_info_total.create({ 26 date: data['date'], 27 time: data['time'], 28 unixTime: data['unixTime'] , 29 totalVideos: data['videos'], 30 totalViews: data['views'], 31 totalComments: data['comments'], 32 }); 33 }).on('end', () => { 34 resolve(); 35 }); 36 }); 37} 38 39setTotalDatas(path).then(() => { 40 Day_info_total.findAll().then((datas) => { 41 // 【問題箇所】全要素に対して処理を行いが、非同期であるため処理が終わる前にループする。 42 for (let i = 1; i < datas.length; i++) { 43  // 処理内容 44 console.log(datas[i]); 45 } 46 }).catch((err) => { 47 console.info('err', err); 48 }); 49});

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

Node
DB操作はsequelizeで行っています。

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

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

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

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

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

guest

回答1

0

ベストアンサー

async/await記法を使えば比較的簡単に処理できます。
下記のようにして、someFunctionをasync関数として実装すれば、処理が終わった後に、for文を抜けるようになります。

for (let i = 1; i < datas.length; i++) { await someFunction(); }

async/awaitに関しては、下記の記事などが分かりやすいです。
https://qiita.com/soarflat/items/1a9613e023200bbebcb3

投稿2020/09/01 17:54

Arahabica

総合スコア209

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問