被我创建的 DAO 层卡住了;在单一情况下工作得很好,但是当需要在事务块中保留几个 bean 实例时,我发现我已经把自己编码到了一个角落。为什么?查看下面的 DAO 创建方法:
def create(e: Entity): Option[Int] =
db.handle withSession { implicit ss: Session=>
catching( mapper.insert(e) ) option match {
case Some(success) => Some(Query(sequenceID))
case None => None
}
}
会话块中发生的查询设置为自动提交,因此我无法将多个持久性操作包装在事务块中。例如,这是处理新成员订阅的简化理解
val result = for{
u <- user.dao.create(ubean)
m <- member.dao.create(mbean)
o <- order.dao.create(obean)
} yield (u,m,o)
result match {
case Some((a,b,c)) => // all good
case _ => // failed, need to rollback here
}
我可以手动执行查询,但这很快就会变得丑陋
db.handle withSession { implicit ss: Session=>
ss.withTransaction {
val result = for{
u <- safe( UserMapper.insert(ubean) )
...
}
def safe(q: Query[_]) =
catching( q ) option match {
case Some(success) => Some(Query(sequenceID))
case None => None
}
}
}
因为我最终会重复错误处理,必须在整个应用程序中提供数据库、会话等,而不是封装在 DAO 层中
任何人都在这里有一些关于如何解决这个问题的明智建议?我真的很喜欢 for comprehension 的简洁性,Scala 摇滚 ;-),想法赞赏!