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

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

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

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

Q&A

解決済

1回答

986閲覧

GASのスプレッドシート 配列 高速化

zoira

総合スコア11

Google Apps Script

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

1グッド

3クリップ

投稿2017/11/07 02:10

###前提・実現したいこと
ここに質問したいことを詳細に書いてください
スプレッドシートにランダム関数を最終行まで(約3000個)、一列に並べるスクリプトを作りたい
###発生している問題・エラーメッセージ
以下のソースコードでは処理はできるのですが、とても遅いです。
配列を用いてもっと高速化する方法を教えて欲しいと思います。

###該当のソースコード Google App Script ここにご自身が実行したソースコードを書いてください function readRandom() { var ss = SpreadsheetApp; var sh = ss.getSheetByName("シート1"); var last_row = sh.getLastRow();     var i     //とりあえずD1から最終行までに乱数を出力     for (i=1;i<last_row;i++){     var rand = Math.random();      // Math.random() は0から0.9999....を出力。      ss.getRange('D2').offset(i-1,0).setValue(rand);      //Offsetでiが増える度にセルを移動      } } ###試したこと 配列を用いてSetValuesを使うこと試みましたが自分自身概念が理解できていないためできなかった ###補足情報(言語/FW/ツール等のバージョンなど) より詳細な情報
ssmxgo👍を押しています

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

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

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

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

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

kei344

2017/11/07 02:12

ご自身で試されたコードを質問文に追記し、「何」が「どのように」わからないのか、コードのどの部分で詰まっているのかなどを具体的に追記されたほうが回答が望めると思います。また、質問文のコードはそれぞれコードブロックで囲んでいただけませんか? ```(バッククオート3つ)で囲み、前後に改行をいれるか、コードを選択して「<code>」ボタンを押すとコードブロックになります。
guest

回答1

0

ベストアンサー

SheetオブジェクトのRangesetValueの扱いかたで劇的に速度が変わります。Google Apps Script はサーバーサイドで動作するので、なるべくサーバー側で動作するようなコードを心がけると良いです。
Class Range

以下のコードは、3つのやり方で1,000行のセルに乱数をセットし、それぞれの経過時間を計る簡単なベンチマークプログラムです。

  1. doRandom1 - 質問者様の提示されたやり方。A1形式でセルを指定してB列に値をセット
  2. doRandom2 - 行ごとに、C列にgetRange(row, column)で取得したRangeに乱数をセット
  3. doRandom3 - 乱数をあらかじめ配列に格納しておき、getRange(row, column rowNum) で取得したRange に配列でD列に一挙にセット。Range.setValuesの使用例。

JavaScript

1function doTest() { 2 try { 3 var ss = SpreadsheetApp.getActiveSpreadsheet(); 4 var sheet = ss.getSheetByName("シート2"); 5 var num = 1000; 6 var elapse 7 8 sheet.clear(); 9 elapse = doRandom1(sheet, num); 10 Browser.msgBox("#1: " + String(elapse)); 11 12 elapse = doRandom2(sheet, num); 13 Browser.msgBox("#2: " + String(elapse)); 14 15 elapse = doRandom3(sheet, num); 16 Browser.msgBox("#3: " + String(elapse)); 17 18 } catch (e) { 19 Logger.log("Exception: " + e); 20 } 21 Browser.msgBox(Logger.getLog()); 22} 23 24// A1 notation 25// B列にデータをセット 26function doRandom1(sheet, num) { 27 var sdt = new Date(); 28 29 for (var i = 1; i <= num; i++) { 30 var rand = Math.random(); 31 sheet.getRange('B1').offset(i - 1, 0).setValue(rand); 32 } 33 var edt = new Date(); 34 var elapse = edt.getTime() - sdt.getTime(); 35 36 return elapse; 37} 38 39// C列にデータをセット 40function doRandom2(sheet, num) { 41 var sdt = new Date(); 42 43 for (var i = 1; i <= num; i++) { 44 var rand = Math.random(); 45 // row, column 46 sheet.getRange(i, 3).setValue(rand); 47 } 48 var edt = new Date(); 49 var elapse = edt.getTime() - sdt.getTime(); 50 51 return elapse; 52} 53 54// D列にデータをセット 55function doRandom3(sheet, num) { 56 var sdt = new Date(); 57 58 var array = []; 59 for (var i = 0; i < num; i++) { 60 var rand = Math.random(); 61 var item = [rand]; 62 array.push(item); 63 } 64 65 // row, column, num 66 var range = sheet.getRange(1, 4, num); 67 // 配列で一挙にセット 68 range.setValues(array); 69 70 var edt = new Date(); 71 var elapse = edt.getTime() - sdt.getTime(); 72 73 return elapse; 74}

下の表は実行結果です。下のものほど速いです。本来なら何回もテストして平均を取るべきですが、何回試しても傾向は変わらないので、その内の1回だけを示します。

Noメソッド経過時間(ミリ秒)備考
1doRandom1199,368m秒1行毎にセルにセットする様子が確認できる
2doRandom21,295m秒スクリプトの終了後にシートに表示される
3doRandom322m秒スクリプトの終了後にシートに表示される

スクリプトの終了後にシートに反映されるのは、サーバー側で処理された後、一挙にクライアント(ブラウザー)側に返されるからだと思います。

参考に、シートの様子を張っておきます。
イメージ説明

投稿2017/11/07 09:50

編集2017/11/07 10:00
dodox86

総合スコア9183

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

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

zoira

2017/11/20 05:15

大変分かりやすい回答ありがとうございました。処理速度が驚くほど違いますね!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問