3

我必须or使用参数值向查询添加条件。

例子:select * from users where email = "abc@xyz.com" or phone="1234123412";

用户可以发送两个字段或只发送一个。我想在每个字段的循环中执行此操作,并在or where条件下添加它们中的每一个。

val query = Users.selectAll()
**var predicates = Op.build { Users.id inList listOf<Int>()}**
for((k, v) in params) {
    val value = URLDecoder.decode(v.first(), "UTF-8")
    predicates = when(k) {
        "email" -> predicates.or(Users.email eq value)
        "phone" -> predicates.or(Users.phone eq value)
    }
}
query.andWhere { predicates }

上面的 DSL 产生下面的 SQL。

SELECT * from users where (((false = true) OR (users.email = "abc@xyz.com")) OR (users.phone = "1234567890"))

看到了false = true吗?那是因为,要使用.or方法,我必须使用条件进行初始化。下面给出的代码片段是添加用于初始化谓词的不必要的代码行。

var predicates = Op.build { Users.id inList listOf<Int>()}

初始化它的正确方法是什么,以便我可以无缝地将多个orand谓词添加到查询中?

4

2 回答 2

4

首先,我建议优化您的代码,例如:

    val params = mapOf<String, List<String>>()
    val emails = params.filterKeys { it == "email" }.map { URLDecoder.decode(it.value.first(), "UTF-8") }
    val phones = params.filterKeys { it == "phone" }.map { URLDecoder.decode(it.value.first(), "UTF-8") }

    if (emails.isEmpty() && phones.isEmpty()) {
        error("No suitable params were provided")
    } else {
        Users.select {
            Users.email inList emails or (Users.phone inList phones)
        }
    }

UPD:orWhere 函数从 0.16.1 版本开始在 Exposed 中可用。

如果您想使用模式匹配 ( when),请以相同的方式定义您的本地orWhere函数andWhere

fun Query.orWhere(andPart: SqlExpressionBuilder.() -> Op<Boolean>) = adjustWhere {
    val expr = Op.build { andPart() }
    if(this == null) expr
    else this or expr
}

并像这样使用它:

val query = Users.selectAll()
for((k, v) in params) {
    val value = URLDecoder.decode(v.first(), "UTF-8")
    predicates = when(k) {
        "email" -> query.orWhere { Users.email eq value }
        "phone" -> query.orWhere{ Users.phone eq value }
    }
}

于 2019-01-26T08:20:02.200 回答
1

一个技巧是初始化predicatesnull定义您自己的orand andon Op<Boolean>?

// not sure x's type is correct here, the wiki doesn't give exact signatures
// may also need @JvmName to allow overloading
fun Op<Boolean>?.or(x: Op<Boolean>) = if (this != null) this.or(x) else x

...
var predicates: Op<Boolean>? = null
...
predicates = when(k) {
    "email" -> predicates.or(Users.email eq value)
    "phone" -> predicates.or(Users.phone eq value)
}

是否值得复杂化,我怀疑:正如 Giacomo 的评论所提到的,数据库肯定会优化您的原始内容false = true

于 2019-01-25T10:57:00.853 回答