ChatGPT解决这个技术问题 Extra ChatGPT

您如何在 Rails 5 ActiveRecord 中执行 or 查询?此外,是否可以在 ActiveRecord 查询中将 orwhere 链接起来?


K
K M Rakibul Islam

Rails 5 将提供在 ActiveRecord 查询中链接 or 子句和 where 子句的功能。请参阅 related discussion and the pull request

因此,您将能够在 Rails 5 中执行以下操作:

要获得具有 id 1 或 2 的 post

Post.where('id = 1').or(Post.where('id = 2'))

其他一些例子:

(A && B) || C:

    Post.where(a).where(b).or(Post.where(c))

(A || B) && C:

    Post.where(a).or(Post.where(b)).where(c)

我怎样才能得到(A || B)&&(C || D)。我试过 Post.where(a).or(Post.where(b)).where(c).or(Post.where(d)) 但它产生为: (A || B) && C || D
@Imran 我相信这将是 Post.where(a).or(Post.where(b)).where(Post.where(c).or(Post.where(d))) 这应该创建 (a || b) && (c || d)
@Imran 这似乎对我不起作用:我得到 ArgumentError: Unsupported argument type: #<MyModel::ActiveRecord_Relation:0x00007f8edbc075a8> (MyModel::ActiveRecord_Relation)
.or 的等价物接受一个关系并产生一个 and 是 .merge。 (A || B) && ( C || D) 可以通过 Post.where(a).or(Post.where(b)).merge(Post.where(c).or(Post.where(d) )))
@MathieuJ。它是 ActiveRecord::Relation#merge。 api.rubyonrails.org/classes/ActiveRecord/…
D
Dipak Gupta

我们不需要等待 rails 5 使用这个 OR 查询。我们也可以将它与 rails 4.2.3 一起使用。有一个反向端口 here

感谢 Eric-Guo 提供 gem where-or,现在我们也可以使用此 gem 在 >= rails 4.2.3 中添加此 OR 功能。


N
Nicholas

(只是对 KM Rakibul Islam 答案的补充。)

使用范围,代码可以变得更漂亮(取决于眼睛看):

scope a,      -> { where(a) }
scope b,      -> { where(b) }

scope a_or_b, -> { a.or(b) }

m
mtrolle

我需要做一个 (A && B) || (C && D) || (E && F)

但是在 Rails 5.1.4 的当前状态下,这太复杂了,无法使用 Arel 或链来完成。但我仍然想使用 Rails 生成尽可能多的查询。

所以我做了一个小技巧:

在我的模型中,我创建了一个名为 sql_whereprivate 方法:

private
  def self.sql_where(*args)
    sql = self.unscoped.where(*args).to_sql
    match = sql.match(/WHERE\s(.*)$/)
    "(#{match[1]})"
  end

接下来在我的范围内,我创建了一个数组来保存 OR

scope :whatever, -> {
  ors = []

  ors << sql_where(A, B)
  ors << sql_where(C, D)
  ors << sql_where(E, F)

  # Now just combine the stumps:
  where(ors.join(' OR '))
}

这将产生预期的查询结果:SELECT * FROM `models` WHERE ((A AND B) OR (C AND D) OR (E AND F))

现在我可以轻松地将它与其他范围等结合起来,而不会出现任何错误的 OR。

美妙之处在于我的 sql_where 采用正常的 where 子句参数:sql_where(name: 'John', role: 'admin') 将生成 (name = 'John' AND role = 'admin')


我认为您可以将 .merge 用作 && 的等价物,并构建一棵适当的树来捕获您的括号。类似于... (scopeA.merge(scopeB)).or(scopeC.merge(scopeD)).or(scopeE.merge(scopeF)),假设每个范围看起来都类似于 Model.where(...)
在使用合并之前检查一下 - github.com/rails/rails/issues/33501
F
Foram

Rails 5 能够使用 where 处理 or 子句。例如。

User.where(name: "abc").or(User.where(name: "abcd"))