FunDA的特点之一是以数据流方式提供逐行数据操作支持。这项功能解决了FRM如Slick数据操作以SQL批次模式为主所产生的问题。为了实现安全高效的数据行操作,我们必须把FRM产生的Query结果集转变成一种强类型的结果集,也就是可以字段名称进行操作的数据行类型结果集。在前面的一篇讨论中我们介绍了通过Shape来改变Slick Query结果行类型。不过这样的转变方式需要编程人员对Slick有较深的了解。更重要的是这种方式太依赖Slick的内部功能了。我们希望FunDA可以支持多种FRM,所以应当尽量避免与任何FRM的紧密耦合。看来从FRM的返回结果开始进行数据行类型格式转换是一种比较现实的选择。一般来说我们还是可以假定任何FRM的使用者对于FRM的Query结果集类型是能理解的,因为他们的主要目的就是为了使用这个结果集。那么由FunDA的使用者提供一个Query结果数据行与另一种类型的类型转换函数应该不算是什么太高的要求吧。FunDA的设计思路是由用户提供一个目标类型以及FRM Query结果数据行到这个强类型行类型的类型转换函数后由FunDA提供强类型行结果集。下面先看一个典型的Slick Query例子:

 import slick.driver.H2Driver.api._
import scala.concurrent.duration._
import scala.concurrent.Await object TypedRow extends App { class AlbumsTable(tag: Tag) extends Table[
(Long,String,String,Option[Int],Int)](tag,"ALBUMS") {
def id = column[Long]("ID",O.PrimaryKey)
def title = column[String]("TITLE")
def artist = column[String]("ARTIST")
def year = column[Option[Int]]("YEAR")
def company = column[Int]("COMPANY")
def * = (id,title,artist,year,company)
}
val albums = TableQuery[AlbumsTable]
class CompanyTable(tag: Tag) extends Table[(Int,String)](tag,"COMPANY") {
def id = column[Int]("ID",O.PrimaryKey)
def name = column[String]("NAME")
def * = (id, name)
}
val companies = TableQuery[CompanyTable] val albumInfo = for {
a <- albums
c <- companies
if (a.company === c.id)
} yield(a.title,a.artist,a.year,c.name) val db = Database.forConfig("h2db") Await.result(db.run(albumInfo.result),Duration.Inf).foreach {r =>
println(s"${r._1} by ${r._2}, ${r._3.getOrElse(2000)} ${r._4}")
} }

上面例子里的albumInfo返回结果行类型是个Tuple类型:(String,String,Option[Int],Int),没有字段名的,所以只能用r._1,r._2...这样的位置注明方式来选择字段。用这种形式来使用返回结果很容易造成混乱,选用字段错误。

前面提到:如果用户能提供一个返回行类型和一个转换函数如下:

   case class AlbumRow(title: String,artist: String,year: Int,studio: String)
def toTypedRow(raw: (String,String,Option[Int],String)):AlbumRow =
AlbumRow(raw._1,raw._2,raw._3.getOrElse(),raw._4)

我们可以在读取数据后用这个函数来转换行类型:

   Await.result(db.run(albumInfo.result),Duration.Inf).map{raw =>
toTypedRow(raw)}.foreach {r =>
println(s"${r.title} by ${r.artist}, ${r.year} ${r.studio}")
}

返回行类型AlbumRow是个强类型。现在我吗可以用字段名来选择数据字段值了。不过,还是有些地方不对劲:应该是用户提供了目标行类型和转换函数后,直接调用一个函数就可以得到需要的结果集了。是的,我们就是要设计一套后台工具库来提供这个函数。

下面我们要设计FunDA的数据行类型class FDADataRow。这个类型现在基本上完全是针对Slick而设的,成功完成功能实现后期再考虑松散耦合问题。这个类型需要一个目标行类型定义和一个类型转换函数,外加一些Slick profile, database等信息。然后提供一个目标行类型结果集函数getTypedRows:

package com.bayakala.funda.rowtypes

import scala.concurrent.duration._
import scala.concurrent.Await
import slick.driver.JdbcProfile object DataRowType {
class FDADataRow[SOURCE, TARGET](slickProfile: JdbcProfile,convert: SOURCE => TARGET){
import slickProfile.api._ def getTypedRows(slickAction: DBIO[Iterable[SOURCE]])(slickDB: Database): Iterable[TARGET] =
Await.result(slickDB.run(slickAction), Duration.Inf).map(raw => convert(raw))
} object FDADataRow {
def apply[SOURCE, TARGET](slickProfile: JdbcProfile, converter: SOURCE => TARGET): FDADataRow[SOURCE, TARGET] =
new FDADataRow[SOURCE, TARGET](slickProfile, converter)
} }

下面是这个函数库的使用示范:

   import com.bayakala.funda.rowtypes.DataRowType

   val loader = FDADataRow(slick.driver.H2Driver, toTypedRow _)

   loader.getTypedRows(albumInfo.result)(db).foreach {r =>
println(s"${r.title} by ${r.artist}, ${r.year} ${r.studio}")
}

那么,作为一种数据行,又如何进行数据字段的更新呢?我们应该把它当作immutable object用函数式方法更新:

   def updateYear(from: AlbumRow): AlbumRow =
AlbumRow(from.title,from.artist,from.year+,from.studio) loader.getTypedRows(albumInfo.result)(db).map(updateYear).foreach {r =>
println(s"${r.title} by ${r.artist}, ${r.year} ${r.studio}")
}

updateYear是个典型的函数式方法:传入AlbumRow,返回新的AlbumRow。

下面是这篇讨论中的源代码:

FunDA函数库:

 package com.bayakala.funda.rowtypes

 import scala.concurrent.duration._
import scala.concurrent.Await
import slick.driver.JdbcProfile object DataRowType {
class FDADataRow[SOURCE, TARGET](slickProfile: JdbcProfile,convert: SOURCE => TARGET){
import slickProfile.api._ def getTypedRows(slickAction: DBIO[Iterable[SOURCE]])(slickDB: Database): Iterable[TARGET] =
Await.result(slickDB.run(slickAction), Duration.Inf).map(raw => convert(raw))
} object FDADataRow {
def apply[SOURCE, TARGET](slickProfile: JdbcProfile, converter: SOURCE => TARGET): FDADataRow[SOURCE, TARGET] =
new FDADataRow[SOURCE, TARGET](slickProfile, converter)
} }

功能测试源代码:

 import slick.driver.H2Driver.api._

 import scala.concurrent.duration._
import scala.concurrent.Await object TypedRow extends App { class AlbumsTable(tag: Tag) extends Table[
(Long,String,String,Option[Int],Int)](tag,"ALBUMS") {
def id = column[Long]("ID",O.PrimaryKey)
def title = column[String]("TITLE")
def artist = column[String]("ARTIST")
def year = column[Option[Int]]("YEAR")
def company = column[Int]("COMPANY")
def * = (id,title,artist,year,company)
}
val albums = TableQuery[AlbumsTable]
class CompanyTable(tag: Tag) extends Table[(Int,String)](tag,"COMPANY") {
def id = column[Int]("ID",O.PrimaryKey)
def name = column[String]("NAME")
def * = (id, name)
}
val companies = TableQuery[CompanyTable] val albumInfo =
for {
a <- albums
c <- companies
if (a.company === c.id)
} yield(a.title,a.artist,a.year,c.name) val db = Database.forConfig("h2db") Await.result(db.run(albumInfo.result),Duration.Inf).foreach {r =>
println(s"${r._1} by ${r._2}, ${r._3.getOrElse(2000)} ${r._4}")
} case class AlbumRow(title: String,artist: String,year: Int,studio: String)
def toTypedRow(raw: (String,String,Option[Int],String)):AlbumRow =
AlbumRow(raw._1,raw._2,raw._3.getOrElse(),raw._4) Await.result(db.run(albumInfo.result),Duration.Inf).map{raw =>
toTypedRow(raw)}.foreach {r =>
println(s"${r.title} by ${r.artist}, ${r.year} ${r.studio}")
} import com.bayakala.funda.rowtypes.DataRowType.FDADataRow val loader = FDADataRow(slick.driver.H2Driver, toTypedRow _) loader.getTypedRows(albumInfo.result)(db).foreach {r =>
println(s"${r.title} by ${r.artist}, ${r.year} ${r.studio}")
} def updateYear(from: AlbumRow): AlbumRow =
AlbumRow(from.title,from.artist,from.year+,from.studio) loader.getTypedRows(albumInfo.result)(db).map(updateYear).foreach {r =>
println(s"${r.title} by ${r.artist}, ${r.year} ${r.studio}")
} }

FunDA(1)- Query Result Row:强类型Query结果行的更多相关文章

  1. [20190214]11g Query Result Cache RC Latches补充.txt

    [20190214]11g Query Result Cache RC Latches补充.txt --//上午测试链接:http://blog.itpub.net/267265/viewspace- ...

  2. [20190214]11g Query Result Cache RC Latches.txt

    [20190214]11g Query Result Cache RC Latches.txt --//昨天我重复链接http://www.pythian.com/blog/oracle-11g-qu ...

  3. django+uwsgi+nginx数据表过大引起"out of memory for query result"

    昨天负责的一个项目突然爆“out of memory for query result”. 背景 项目的数据表是保存超过10m的文本数据,通过json方式保存进postgres中,上传一个13m的大文 ...

  4. PL/SQL:these query result are not updateable,include the ROWID to get updateab -----for update

    these query result are not updateable,include the ROWID to get updateab 原因: 其实,选中一个表后,右键,如果选择“query ...

  5. Oralce查询后修改数据,弹窗报提示these query result are not updateable,include the ROWID to get updateable

    select t.*, (select a.ANNEXNAME from base_annex a where a.id = t.closeFile) closeFileName, (select a ...

  6. java.sql.SQLException: Column count doesn't match value count at row 1 Query: insert into category values(null,?,?,?) Parameters: [1111111, 1111, 软件]

    java.sql.SQLException 问题: java.sql.SQLException: Column count doesn't match value count at row 1 Que ...

  7. 在PL/SQL DEV里面有把锁一样的按钮,点击它会跳出“these query result are not updateable,include the ROWID to get updateab

    在PL/SQL DEV里面有把锁一样的按钮,点击它会跳出“these query result are not updateable,include the ROWID to get updateab ...

  8. 九、MySQL报错( (1292, u"Truncated incorrect DOUBLE value: '424a000000066'") result = self._query(query))

    1.数据库sql语句:SELECT seat_id FROM netsale_order_seat os join netsale_order nor on os.order_code=nor.ord ...

  9. Query DSL for elasticsearch Query

    Query DSL Query DSL (资料来自: http://www.elasticsearch.cn/guide/reference/query-dsl/) http://elasticsea ...

随机推荐

  1. BPM配置故事之案例10-获取外部数据

    老李:Hi,小明,我又来了 小明:--这次又怎么了. 老李:之前的物资管理方式太混乱了,这段时间我整理了采购物资清单,现在都录入到我们的ERP中了,以后申请物资改成从ERP数据选择吧.物资明细表我也做 ...

  2. 安装并使用PHPunit

    安装并使用PHPunit Linux 下安装PHPunit PHP 档案包 (PHAR)  要获取 PHPUnit,最简单的方法是下载 PHPUnit 的 PHP 档案包 (PHAR),它将 PHPU ...

  3. centos 6.5 升级php

    1>追加CentOS 6.5的epel及remi源. # rpm -Uvh http://ftp.iij.ad.jp/pub/linux/fedora/epel/6/x86_64/epel-re ...

  4. Photoshop、Illustrator思维导图笔记

    半年前学习Photoshop时记得的思维导图笔记,可能不是很全,常用的基本都记下了.

  5. Mono 4.0 Mac上运行asp.net mvc 5.2.3

    Mono 4.0 已经发布,二进制包已经准备好,具体的发布说明参见:http://www.mono-project.com/docs/about-mono/releases/4.0.0/. 今天在Ma ...

  6. Netty构建分布式消息队列实现原理浅析

    在本人的上一篇博客文章:Netty构建分布式消息队列(AvatarMQ)设计指南之架构篇 中,重点向大家介绍了AvatarMQ主要构成模块以及目前存在的优缺点.最后以一个生产者.消费者传递消息的例子, ...

  7. Lesson 16 A polite request

    Text If you park your car in the wrong place, a traffic policeman will soon find it. You will be ver ...

  8. 虚拟机体验之 KVM 篇

    在上一篇中,我展示了虚拟机软件 QEMU 的使用.效果及其性能,同时也分析了不同用户对虚拟机的不同追求.但是不管是桌面用户还是企业级用户,对虚拟机软件的追求有一点是共同的,那就是性能.QEMU 是一个 ...

  9. 利用HTML5 的Datalist 元素实现输入提示

    HTML5有无限可能,总是在释出一些新鲜实用的功能,让原生的web环境更加炫酷. 今天看到datalist 这个元素,可以用来预先定义一个输入框的潜在选项,也就是我们在平时项目中经常用jQuery插件 ...

  10. rsync 笔记之 list

    通过 rsync --list-only 可以列出可用的 文件/目录或者 module 下面两者的含义是完全不同的: rsync --list-only root@192.168.4.140: 使用系 ...