scala - Slick for-comprehensions:如何处理可以为 None 或 Some-thing 的枚举数?
问题描述
我有一个用例可能不是特定于 Slick,而是用于理解。高级模式是这样的,但它会导致 enum3 中的编译器错误:
(for {
enum1 <- // ...
enum2 <- // ...
enum3 <- optionArg.fold(empty result).map { ... }
} yield ())
然后我最终做了以下编译但有代码重复的代码,即 enum1 和 enum2:
optionArg match {
case (Some(arg)) => {
(for {
enum1 <- // ...
enum2 <- // ...
enum3 <- // do something with arg
} yield ())
}
case None => {
(for {
enum1 <- // ...
enum2 <- // ...
} yield ())
}
}
}
具体来说,我有这个重复的变体编译:
/**
* Returns the inserted `oauth2Info` instance including the params. We first
* look up the `LoginInfo` by the relevant search criteria, fetching its `userId`
* which is then used to persist a `OAuth2Info` and multiple `OAuth2InfoParam`.
*
* @param extLoginInfo The login info for which the auth info should be added.
* @param extOAuth2Info The TOTP info to add containing the params.
* @return the inserted `oauth2Info` instance including the params.
*/
def add(extLoginInfo: ExtLoginInfo, extOAuth2Info: ExtOAuth2Info): Future[ExtOAuth2Info] = {
val insertion = extOAuth2Info.params match {
case Some(params) => {
(for {
userId <- LoginInfo.filter { loginInfo => loginInfo.providerId === extLoginInfo.providerID && loginInfo.providerKey === extLoginInfo.providerKey }.map(_.userId).result.head
_ <- (OAuth2Info += OAuth2InfoRow(userId, extOAuth2Info.accessToken, extOAuth2Info.tokenType, extOAuth2Info.expiresIn, extOAuth2Info.refreshToken))
_ <- DBIOAction.sequence(params.map { param => (OAuth2InfoParam += OAuth2InfoParamRow(userId, param._1, param._2)) })
} yield ())
}
case None => {
(for {
userId <- LoginInfo.filter { loginInfo => loginInfo.providerId === extLoginInfo.providerID && loginInfo.providerKey === extLoginInfo.providerKey }.map(_.userId).result.head
_ <- (OAuth2Info += OAuth2InfoRow(userId, extOAuth2Info.accessToken, extOAuth2Info.tokenType, extOAuth2Info.expiresIn, extOAuth2Info.refreshToken))
} yield ())
}
}
db.run(insertion.transactionally).map(_ => extOAuth2Info)
}
而这个简洁的所需变体不会:
def add(extLoginInfo: ExtLoginInfo, extOAuth2Info: ExtOAuth2Info): Future[ExtOAuth2Info] = {
val insertion = (for {
userId <- LoginInfo.filter { loginInfo => loginInfo.providerId === extLoginInfo.providerID && loginInfo.providerKey === extLoginInfo.providerKey }.map(_.userId).result.head
_ <- (OAuth2Info += OAuth2InfoRow(userId, extOAuth2Info.accessToken, extOAuth2Info.tokenType, extOAuth2Info.expiresIn, extOAuth2Info.refreshToken))
_ <- extOAuth2Info.params.fold(DBIOAction.seq()) { params =>
DBIOAction.sequence(params.map { param => (OAuth2InfoParam += OAuth2InfoParamRow(userId, param._1, param._2)) })
}
} yield ()).transactionally
db.run(insertion).map(_ => extOAuth2Info)
}
给出编译错误:
[play-silhouette-seed] $ compile
[info] Formatting 1 Scala source ProjectRef(uri("file:/home/skywalker/code/play-silhouette-seed/"), "root")(compile) ...
[info] Compiling 1 Scala source to /home/skywalker/code/play-silhouette-seed/target/scala-2.12/classes ...
[error] /home/skywalker/code/play-silhouette-seed/app/models/daos/OAuth2InfoDaoImpl.scala:58:28: type mismatch;
[error] found : slick.dbio.DBIOAction[scala.collection.immutable.Iterable[Int],slick.dbio.NoStream,slick.dbio.Effect.Write]
[error] required: slick.dbio.DBIOAction[Unit,slick.dbio.NoStream,slick.dbio.Effect]
[error] DBIOAction.sequence(params.map { param => (OAuth2InfoParam += OAuth2InfoParamRow(userId, param._1, param._2)) })
[error] ^
[error] one error found
[error] (Compile / compileIncremental) Compilation failed
[error] Total time: 1 s, completed May 28, 2019 10:42:00 AM
解决方案
fold
当编译器无法对表达式的正确泛型类型进行类型推断时,这是一个问题。它没有正确地将第一个表达式返回的类型扩展为第二个表达式的类型。你需要帮助编译器一点。
在这样的情况下也会发生
Some(123).fold(None)(_ => Some("123"))
// type mismatch; found: Some[String] required: None.type
您可以更改fold
为map().getOrElse()
(然后“好类型”首先出现并被推断)。
或者您可以在某处添加类型注释,例如
private val noAction: slick.dbio.DBIOAction[Iterable[Int],NoStream,Effect.Write]
= DBIOAction.sequence()
theOption.fold(noAction)(params => .... )
此外,如果您发现自己在重复代码,可以通过移出通用表达式来缓解这种情况。使用像 Slick 这样的功能性(无副作用)代码,这种重构是非常安全的。您可以只构建几个val
或def
使用您需要运行的 Slick 操作,然后再组合它们。
def add(extLoginInfo: ExtLoginInfo, extOAuth2Info: ExtOAuth2Info): Future[ExtOAuth2Info] = {
val getUserId = LoginInfo.filter { loginInfo => loginInfo.providerId === extLoginInfo.providerID && loginInfo.providerKey === extLoginInfo.providerKey }.map(_.userId).result.head
val insertOAuth = for {
userId <- getUserId
_ = (OAuth2Info += OAuth2InfoRow(userId, extOAuth2Info.accessToken, extOAuth2Info.tokenType, extOAuth2Info.expiresIn, extOAuth2Info.refreshToken))
} yield userId
val insertion = extOAuth2Info.params match {
case Some(params) => {
(for {
userId <- insertOAuth
_ <- DBIOAction.sequence(params.map { param => (OAuth2InfoParam += OAuth2InfoParamRow(userId, param._1, param._2)) })
} yield ())
}
case None =>
insertOAuth
}
db.run(insertion.transactionally).map(_ => extOAuth2Info)
}
</p>
推荐阅读
- regex - 如何使用反向引用来替换部分字符串?
- python - Python中数组的大小与变量的大小
- python-3.x - 从 S3 读取 JSON 文件的 Dask 列表索引超出范围
- python - 可序列化文件对象
- python - 尝试将实例变量映射到类中的字典
- c++ - 制作字母金字塔时出现大量字符错误
- sql - 如何根据另一列 PROC SQL 的条件创建新列
- c++ - 在我的 C++ 的 vscode 设置中出现错误
- flutter - 底部导航栏停止工作 ScrollController 附加到多个滚动视图 pagecontroller.jumpTo 错误
- google-chrome-devtools - 如何使用 CacheStorage 导出保存在浏览器中的缓存文件?