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

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

新規登録して質問してみよう
ただいま回答率
85.35%
Ruby on Rails

Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

Q&A

解決済

1回答

526閲覧

Rails 処理でリセットする場合、ロールバックを使ってもよいでしょうか

orori

総合スコア42

Ruby on Rails

Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

0グッド

0クリップ

投稿2021/08/31 22:17

Rails6、 MySql8で先着順の商品購入サイトをつくっています。

同一の商品(item)を複数のユーザーが購入できないようにしています。
購入の際、悲観的ロックをかけて、itemのstatusがavailableのものだけを購入できるようにしています。
もし、ほかのひとが購入などしてしまって、itemのstatusが、availableではなくなった場合は、
else以降で、changedに追加したもともとの値をつかってリセットし、breakしています。

実際には、購入時は、関連テーブルの処理など複雑になりますが、
これを例えば、else以降で、raiseでロールバックを起こして、処理をリセットさせるほうが
コードを書く際は簡単なのですが、これは行わない方がよいでしょうか。

確か処理が遅くなるという記述をどこかでみたような気がするのですが、
よろしくお願いいたします。

ruby

1def update_items(item_ids:) 2 changed = [] 3 results = { success: true, count: 0, price: 0 } 4 Item.transaction do 5 item_ids.each do |item_id| 6 item = Item.lock.find(item_id) 7 if item.status == "available" 8 changed << ticket.dup 9 results[:count] += 1 10 results[:price] += t.price 11 item.update!(status: "purchased") 12 else 13 changed.each do |c| 14 item = Item.lock.find(c.id) 15 item.update!(status: c.status) 16 end 17 results[:success] = false 18 break 19 end 20 end 21 end 22 results 23end

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

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

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

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

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

guest

回答1

0

ベストアンサー

例外を起こして ロールバックというのが(速度はわかりませんが)一般的ですね。
transactionという仕掛けが用意されています。

投稿2021/08/31 22:31

winterboum

総合スコア23567

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

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

orori

2021/09/01 04:11 編集

なるほど、通常はそのようにされているのですね! お聞きしたいのですが、エラーを補足して、DBの現状を返す場合、 例えば、コントローラは以下のようになりますでしょうか? def update Item.transaction do (処理) @item = Item.all     @message = ''商品を購入しました' end rescue ActiveRecord::RecordInvalid @item = Item.all(処理されてない現状を返す) @message = "該当の商品は既に購入されました" end すみません、いろいろやり方はあるかと思いますが、 よろしくお願いいたします。
orori

2021/09/01 04:12 編集

rescue ActiveRecord::RecordInvalidだと、補足されないですね・・・ 例外はtransactionの外側では捕捉されないので、 transaction内部でsuccessなどのパラメータをつくって、それを用いるしかないようですね。 ありがとうございました!!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問