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. Ajax使用WCF实现小票pos机打印源码

    通过ajax跨域方式调用WCF服务,实现小票pos机的打印,源码提供web方式,客户端方式测试,服务驻留右侧底部任务栏,可控制服务开启暂停,用户可自定义小票打印模板,配合零售录入. qq  22945 ...

  2. IOS开发基础知识--碎片51

    1:https关闭证书跟域名的验证 AFSecurityPolicy *securityPolicy = [AFSecurityPolicy defaultPolicy]; securityPolic ...

  3. mono for android 读取网络远程图片

    布局 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android=& ...

  4. NodeJs 开发微信公众号(四)微信网页授权

    微信的网页授权指的是在微信公众号中访问第三方网页时获取用户地理.个人等信息的权限.对于开发了自己的网页app应用时,获取个人的信息非常重要.上篇博客讲到了注册时可以获取用户的信息,很多人会问为什么还需 ...

  5. Running Dubbo On Spring Boot

    Dubbo(http://dubbo.io/) 是阿里的开源的一款分布式服务框架.而Spring Boot则是Spring社区这两年致力于打造的简化Java配置的微服务框架. 利用他们各自优势,配置到 ...

  6. TODO:一不顺眼就换字体Go之应用篇

    TODO:一不顺眼就换字体Go之应用篇 字体,文字的外在形式特征.就是文字的风格,是文字的外衣.好的字体让人看得舒服,让人更有看.写的欲望哦.2016-11-16 GO官方博客发布了同名Go字体,并没 ...

  7. 随笔jade

    mixin在刚使用的时候写错,写成了minxin,然后进行命令生成,发现报错 查了蛮久,由于开发工具并没有提示错误,最后找到了这样的错误,记下来,望大家不要重复爬坑

  8. 浅析Java 泛型

    泛型是JavaSE5引入的一个新概念,但是这个概念在编程语言中却是很普遍的一个概念.下面,根据以下内容,我们总结下在Java中使用泛型. 泛型使用的意义 什么是泛型 泛型类 泛型方法 泛型接口 泛型擦 ...

  9. 如何开发一款堪比APP的微信小程序(腾讯内部团队分享)

    一夜之间,微信小程序刷爆了行业网站和朋友圈,小程序真的能如张小龙所说让用户"即用即走"吗? 其功能能和动辄几十兆安装文件的APP相比吗? 开发小程序,是不是意味着移动应用开发的一次 ...

  10. eclipse内下载及配置maven插件(转)

    本文介绍Maven的安装和配置,同样适用于eclipse 1.首先需要安装jdk,eclipse(废话!). 然后到maven官网下载maven,http://maven.apache.org/dow ...