如何在 ARel 中使用 CONCAT() 等函数?

Posted

技术标签:

【中文标题】如何在 ARel 中使用 CONCAT() 等函数?【英文标题】:How do I use functions like CONCAT(), etc. in ARel? 【发布时间】:2012-03-07 11:08:20 【问题描述】:

有没有办法让ARel 将列名写入(已清理,可能有别名等)CONCAT() 和其他 SQL 函数?

这里是how to do it with AVG()...

?> name = Arel::Attribute.new(Arel::Table.new(:countries), :name)
=> #<struct Arel::Attributes::Attribute [...]
?> population = Arel::Attribute.new(Arel::Table.new(:countries), :population)
=> #<struct Arel::Attributes::Attribute [...]
?> Country.select([name, population.average]).to_sql
=> "SELECT `countries`.`name`, AVG(`countries`.`population`) AS avg_id FROM `countries`"

(是的,我知道avg_id 在每一行都是一样的,只是想说明我的问题)

如果我想要一个不同的功能呢?

?> Country.select(xyz).to_sql # Arel::Concat.new(name, population) or something?
=> "SELECT CONCAT(`countries`.`name`, ' ', `countries`.`population`) AS concat_id FROM `countries`"

谢谢!

【问题讨论】:

Sequel 有一种将列选择为 "...".lit 的方法,意思是“文字 SQL”,而不是在这种情况下被解释为字符串。这会禁用 SQL 转义,因此您可以注入任何您想要的东西。不确定 AREL 等价物是什么,但也许这是一个想法。 我自己在这里写了更多细节mrpunkin.com/post/18919379925/using-arel-for-sql-functions> 【参考方案1】:

使用NamedFunction:

name = Arel::Attribute.new(Arel::Table.new(:countries), :name)
func = Arel::Nodes::NamedFunction.new 'zomg', [name]
Country.select([name, func]).to_sql

【讨论】:

看起来 NamedFunction 需要附加到属性?例如如果我想按日期对事物进行分组,我可以做.group(arel_table[:created_at].date),而不直接在“created_at”上附加“日期”函数,以便我可以在其他地方重用它吗? (sql:GROUP BY DATE(created_at)) @AaronPatterson 使用您的代码,func.to_sql 给出zomg('countries', 'name')。我想,代码应该是...Function.new 'zomg', [name](带方括号)。【参考方案2】:

您还可以使用Arel Extensions gem 来更简单地访问函数。

> User.where((User[:login] + User[:first_name]).length.in 2..10).to_sql
"SELECT `users`.* FROM `users` 
 WHERE LENGTH(CONCAT(CAST(`users`.`login` AS char),
                     CAST(`users`.`first_name` AS char)))
         BETWEEN (2) AND (10)"

【讨论】:

以上是关于如何在 ARel 中使用 CONCAT() 等函数?的主要内容,如果未能解决你的问题,请参考以下文章

Arel:如何在 arel speak 中写 attr = NULL?

如何在初始 where 语句中使用 Arel::Nodes::TableAlias

如何在 Arel 中“存在于哪里”

如何在 Rails 中对 AREL 中的子查询进行连接

如何在 Arel 和 Rails 中进行 LIKE 查询?

如何使用 arel/关系代数获取不同的值