ScalaQuery and Play框架:处理未分配(AutoInc)主键的最佳实践



我使用的是带有ScalaQuery 0.9.5的Play 2.0.2。

我有以下简单的型号代码:

case class Task(id: Long, name: String)
object Task extends Table[(Long, String)]("task") {
lazy val database = Database.forDataSource(DB.getDataSource())
def id = column[Long]("id", O PrimaryKey, O AutoInc)
def name = column[String]("name", O NotNull)
def create(task: Task) = database.withSession {
  implicit db: Session => {
    Task.name insert(task.name)
  }
}

以及以下处理表单提交的代码:

val taskForm: Form[Task] = Form(
  mapping(
    "name" -> nonEmptyText
  ) {
    (name) => Task(-1L, name)
  } {
    task => Some(task.name)
  }
)
def newTask = Action {
  implicit request =>
    taskForm.bindFromRequest.fold(
      errors => BadRequest(views.html.index(Task.all, errors)),
      task => {
        Task.create(task)
        Redirect(routes.Application.tasks())
      }
    )
}

几个问题:

1) 有没有比传递常量更好的方法来处理瞬态主键值?类似于Anorm的NotAssigned?

2) 将"id"->gnored(-1L)添加到Form映射并使用Task的提取器函数会更好吗?

3) 应该在没有id字段的情况下定义case类吗?

如果Task有id,则不应使用此类来表示(尚未)没有id的值。无论如何,通常情况下,创建的对象包含的数据与表单中填写的数据不同(例如创建日期等),因此使用同一类来表示创建对象和创建的对象本身所需的值通常是没有意义的。

因此,您可以定义另一个case类,例如TaskForm(它最终可能继承自Task的公共基),或者,在您的情况下,只使用产生String值的形式:

// Task
case class Task(id: Long, name: String)
object Task extends Table[(Long, String)]("task") {
  lazy val database = Database.forDataSource(DB.getDataSource())
  def id = column[Long]("id", O PrimaryKey, O AutoInc)
  def name = column[String]("name", O NotNull)
  def create(name: String) = database.withSession { implicit db: Session =>
    Task.name insert(name)
  }
}

// Application
object Application extends Controller {
  val taskForm = Form("name" -> nonEmptyText)
  def newTask = Action { implicit request =>
    taskForm.bindFromRequest.fold(
      errors => BadRequest(views.html.index(Task.all, errors)),
      name => {
        Task.create(name)
        Redirect(routes.Application.tasks())
      }
    )
  }
}

最新更新