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

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

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

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

Q&A

解決済

2回答

703閲覧

いいね機能においてN+1問題を解決したい

退会済みユーザー

退会済みユーザー

総合スコア0

Ruby on Rails

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

0グッド

0クリップ

投稿2018/03/30 08:41

いつもお世話になっています。
Railsでいいね機能を実装したのですが、いいねの部分でN+1問題が解決できないので、質問致します。

現在は

  • Userモデル
  • Tweetモデル
  • Likeモデル

の3つのモデルを作っており、User→Tweet→Likeのように関連付けをしています。

そして、tweets_controller.rbは以下の様に書いています。

//tweets_controller.rb def index @tweets = Tweet.all.includes(:user) end

そして、tweets/index内に以下の記述をしていて、

<% @tweets.each do |tweet| %> 省略 <%= render partial:'likes/like',locals:{tweet:tweet} %> <% end %>

likes/_like.html.erb内には、以下の様に書いています。

<% if user_signed_in? %> <% if tweet.likes.find_by(user_id:current_user.id) %> //この行がN+1問題の原因? <%= button_to [like, tweet.likes.find_by(user_id:current_user.id)], method: :delete do %> いいね済 <% end %> <% else %> <%= button_to tweet_likes_path(tweet) do %> いいね <% end %> <% end %> <% else %> ログインするといいねができます。 <% end %>

ログを見ると、おそらく2行目の「<% if tweet.likes.find_by(user_id:current_user.id) %>」がN+1問題を引き起こしていると思うのですが、何か解決策はあるのでしょうか?

よろしくお願い致します。

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

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

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

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

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

guest

回答2

0

ベストアンサー

find_byによるN+1問題の解決法が分からないので、妥協案として以下の方法を採用しました。

https://qiita.com/strviola/items/5e509b2102556eb7c082の記事がヒントになりそうですが、まだ理解ができていないです)

まず、いいねの部分テンプレートである、likes/_like.html.erbを以下の様に訂正しました。

<% if user_signed_in? %> <%= button_to tweet_likes_path(tweet) do %> いいね <% end %> <% else %> ログインするといいねができます。 <% end %>

find_byメソッドを廃止して、「いいねをしていない場合はいいねボタン(create)を表示して、いいねをしている場合は、いいね済み(destroy)と表示させるのを止めました。

そして、likes_contorollerに以下を追加しました。

def create @tweet = Tweet.find(params[:tweet_id]) return if @tweet.likes.find_by(user_id:current_user.id)//この一行を追加 like.create(省略 end

これで一人一回いいねができる、と言う実装をしました。

投稿2018/03/31 04:15

退会済みユーザー

退会済みユーザー

総合スコア0

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

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

0

おそらく、LikeをロードしていないためN+1問題が起きているのではと思います。
UserをロードするタイミングでLikeも一緒にロードしてあげれば解決するかと。

@tweets = Tweet.all.includes([:user, :like])

投稿2018/03/30 09:48

KaiShoya

総合スコア551

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

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

退会済みユーザー

退会済みユーザー

2018/03/31 02:43

ご回答ありがとうございます。 試しにやってみましたが、N+1問題を解決できませんでした。 やはり、find_byが呼ばれる毎に、クエリが発生している気がします。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問