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. 值得注意的ibatis动态sql语法格式

    一.Ibatis常用动态sql语法,简单粗暴用一例子 <select id="iBatisSelectList" parameterClass="java.util ...

  2. 免费公开课,讲解强大的文档集成组件Aspose,现在可报名

    课程①:Aspose.Total公开课内容:讲解全能型文档管理工具Aspose.Total主要功能及应用领域时间:2016-11-24 14:30 (暂定)报名地址:http://training.e ...

  3. DB2重启数据库实例

    DB2重启数据库实例时,有时停止实例会失败,此时需要先确认没有应用链接数据库,然后再关闭数据库实例,并重新启动. 1.查看是否有活动的链接 命令:db2 list applications for d ...

  4. 编写简单的Makefile文件

    makefile中的编写内容如下: www:hello.c x.h gcc hello.c -o hello clean: rm hello www:hello.c  x.h 表示生成www这个文件需 ...

  5. Hello Jexus

    一.关于 CentOS CentOS(Community Enterprise Operating System)是Linux发行版之一,它是来自于Red Hat Enterprise Linux依照 ...

  6. Xamarin.Android-用ZXing实现二维码扫描以及连续扫描

    一.前言 本文的内容有两个基础:ZXing.Net和ZXing.Net.Mobile ZXing.Net:ZXing的C#实现,主要封装了各种二维码的编码.解码等跨平台的算法 ZXing.Net.Mo ...

  7. 剁手党也有春天 -- 淘宝 UWP ”比较“功能诞生记

    前言 网购已经不再是现在的时髦,而变成了我们每天的日常生活.上网已经和买买买紧密地联系在了一起,成为了我们的人生信条.而逛街一词,越来越多地变成了一种情怀.有时候我们去逛街,要么是为了打发时间,要么是 ...

  8. SQL Server 索引和表体系结构(聚集索引)

    聚集索引 概述 关于索引和表体系结构的概念一直都是讨论比较多的话题,其中表的各种存储形式是讨论的重点,在各个网站上面也有很多关于这方面写的不错的文章,我写这篇文章的目的也是为了将所有的知识点尽可能的组 ...

  9. Struts2+Spring+Hibernate框架整合总结详细教程

    一.SSH三大框架知识总结 Struts 2是Struts的下一代产品,是在 struts 1和WebWork的技术基础上进行了合并的全新的Struts 2框架.其全新的Struts 2的体系结构与S ...

  10. Android View.setId(int id) 用法

    Android View.setId(int id) 用法 当要在代码中动态的添加View并且为其设置id时,如果直接用一个int值时,Studio会警告. 经过查询,动态设置id的方法有两种; 1. ...