gpt4 book ai didi

scala - 如何在 select 子句中编写嵌套查询

转载 作者:行者123 更新时间:2023-12-04 02:07:53 26 4
gpt4 key购买 nike

我正在尝试使用 SLICK 1.0.0 生成此 SQL:

    select
cat.categoryId,
cat.title,
(
select
count(product.productId)
from
products product
right join products_categories productCategory on productCategory.productId = product.productId
right join categories c on c.categoryId = productCategory.categoryId
where
c.leftValue >= cat.leftValue and
c.rightValue <= cat.rightValue
) as productCount
from
categories cat
where
cat.parentCategoryId = 2;

我最成功的尝试是(我删除了“连接”部分,因此更具可读性):
def subQuery(c: CategoriesTable.type) = (for {
p <- ProductsTable

} yield(p.id.count))
for {
c <- CategoriesTable
if (c.parentId === 2)
} yield(c.id, c.title, (subQuery(c).asColumn))

这会在子查询中生成缺少括号​​的 SQL:
   select 
x2.categoryId,
x2.title,
select count(x3.productId) from products x3
from
categories x2
where x2.parentCategoryId = 2

这显然是无效的 SQL
任何想法如何让 SLICK 将这些括号放在正确的位置?或者也许有不同的方法来实现这一目标?

最佳答案

我从未使用过 Slick 或 ScalaQuery,因此找出如何实现这一点是一次冒险。 Slick 是非常可扩展的,但是关于扩展的文档有点棘手。它可能已经存在,但这是我想出的。如果我做错了什么,请纠正我。

首先,我们需要创建一个自定义驱动程序。我扩展了H2Driver以便能够轻松测试。

trait CustomDriver extends H2Driver {

// make sure we create our query builder
override def createQueryBuilder(input: QueryBuilderInput): QueryBuilder =
new QueryBuilder(input)

// extend the H2 query builder
class QueryBuilder(input: QueryBuilderInput) extends super.QueryBuilder(input) {

// we override the expr method in order to support the 'As' function
override def expr(n: Node, skipParens: Boolean = false) = n match {

// if we match our function we simply build the appropriate query
case CustomDriver.As(column, LiteralNode(name: String)) =>
b"("
super.expr(column, skipParens)
b") as ${name}"

// we don't know how to handle this, so let super hanle it
case _ => super.expr(n, skipParens)
}
}
}

object CustomDriver extends CustomDriver {
// simply define 'As' as a function symbol
val As = new FunctionSymbol("As")

// we override SimpleSql to add an extra implicit
trait SimpleQL extends super.SimpleQL {

// This is the part that makes it easy to use on queries. It's an enrichment class.
implicit class RichQuery[T: TypeMapper](q: Query[Column[T], T]) {

// here we redirect our as call to the As method we defined in our custom driver
def as(name: String) =
CustomDriver.As.column[T](Node(q.unpackable.value), name)
}
}

// we need to override simple to use our version
override val simple: SimpleQL = new SimpleQL {}
}

为了使用它,我们需要导入特定的东西:
import CustomDriver.simple._
import Database.threadLocalSession

然后,要使用它,您可以执行以下操作(我在示例中使用了官方 Slick 文档中的表格)。
// first create a function to create a count query
def countCoffees(supID: Column[Int]) =
for {
c <- Coffees
if (c.supID === supID)
} yield (c.length)

// create the query to combine name and count
val coffeesPerSupplier =
for {
s <- Suppliers
} yield (s.name, countCoffees(s.id) as "test")

// print out the name and count
coffeesPerSupplier foreach { case (name, count) =>
println(s"$name has $count type(s) of coffee")
}

结果是这样的:
Acme, Inc. has 2 type(s) of coffee
Superior Coffee has 2 type(s) of coffee
The High Ground has 1 type(s) of coffee

关于scala - 如何在 select 子句中编写嵌套查询,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14920153/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com