阅读更多

5顶
0踩

编程语言
原文地址:https://dzone.com/articles/java-8-query-databases-using-streams
译文转自OneAPM,由OneAPM工程师翻译而成。

Speedment 是使用 ORM 方式操作数据库的一种选择,以前我们需要100行操作数据库的 Java 代码,在 Java 8中,可能只需要一行代码。

在90年代末,我使用 Java 开发数据库应用的时候,许多代码逻辑都需要自己来编写,比如捕获异常、类型转换等,经过许多改动,最后这些代码变得难以维护和扩展。



由于关系型数据库操作语言和面向对象语言之间的差异,如今我们仍然需要花费许多时间建立数据库与 Java 应用之间互相沟通的桥梁。通常,我们可以编写自己的映射层(mapping layer),或者使用第三方的 ORM(Object Relational Mapper)对象关系映射框架,比如 Hibernate。ORM 框架虽然使用起来很方便,但是如何正确地配置和提高框架操作数据库的性能却不太容易,ORM 框架往往会使我们的应用性能下降。

最近,我贡献了一个新的开源项目——Speedment,它能使我们使用 Java 8 开发数据库应用程序变得更为快捷和高效。
Speedment 是什么?
Speedment 是一个开源项目,它是一个基于Java 8的新特性开发的新的 Java 库,从这个项目开发开始,它的代码就全部使用 Java 8来编写。Speedment 使用标准流查询数据库,这使得开发者不需要学习任何新的查询 API ,以及不必考虑 JDBC 、ResultSet 和其他有关数据库的指定的操作。

Speedment 会根据现有数据库来自动生成代码。由于它的这种方式,开发者不需要编写一行关于数据库实体(database entities)的代码。它生成的代码中也包含 JavaDocs 帮助文档,这使开发者不需要编写关于 User 或者 Book 等对象的实体类。取而代之地,我们只需要创建或者使用一个现有的数据库,然后用 Speedment 去连接它,接着 Speedment 会分析数据库结构来生成实体类的代码。

更有趣的是,Speedment 用野兔来作为它的吉祥物。



在接下来的例子中,我们会使用一个名为 “hare” 的数据库来给大家演示 Speedment 的使用方式。该数据库的表结构如下:
mysql> explain hare;  
+-------+-------------+------+-----+---------+----------------+
| Field | Type        | Null | Key | Default | Extra          |
+-------+-------------+------+-----+---------+----------------+
| id    | int(11)     | NO   | PRI | NULL    | auto_increment |
| name  | varchar(45) | NO   |     | NULL    |                |
| color | varchar(45) | NO   |     | NULL    |                |
| age   | int(11)     | NO   |     | NULL    |                |
+-------+-------------+------+-----+---------+----------------+
4 rows in set (0.01 sec)  

下面是 Speedment 根据数据库信息生成的一个相应的实体类(为简洁起见,我们将 JavaDocs 在这里移除了):
public interface Hare extends Entity<Hare> {  
    public final static ReferenceComparableField<Hare, Integer> ID = new ReferenceComparableFieldImpl<>("id", Hare::getId, Hare::setId);
    public final static ReferenceComparableStringField<Hare> NAME = new ReferenceComparableStringFieldImpl<>("name", Hare::getName, Hare::setName);
    public final static ReferenceComparableStringField<Hare> COLOR = new ReferenceComparableStringFieldImpl<>("color", Hare::getColor, Hare::setColor);
    public final static ReferenceComparableField<Hare, Integer> AGE = new ReferenceComparableFieldImpl<>("age", Hare::getAge, Hare::setAge);
    Integer getId();
    String getName();
    String getColor();
    Integer getAge();
    Hare setId(Integer id);
    Hare setName(String name);
    Hare setColor(String color);
    Hare setAge(Integer age);
    /** Graph-like traversal methods eliminating JOINs */
    Stream<Carrot> findCarrotsByOwner();
    Stream<Carrot> findCarrotsByRival();
    Stream<Carrot> findCarrots();
}

我将用一篇单独的文章介绍 find*() 方法的用法,它可以被用来代替 SQL joins 操作。

Queries查询示例

下面的例子展示如何查询 Hare 表的数据库信息:
List<Hare> oldHares = hares.stream()  
    .filter(AGE.greaterThan(8))
    .collect(toList());

智能流
上面的代码看起来已经遍历了 hare 数据库表的所有行,但实际上并不是这样的。 Stream 是智能的,当它到达 collect() 操作的时候,会分析 filter 操作,并推断出 hare.age 大于8的列,因此会节省 hares 的流操作,产生与 “select * from hare where age > 8” 操作一样的效果。如果你使用了多个 filters,他们会被合并起来以节省流操作。下面是另一种用流方式进行多个操作的例子:
long noOldHares = hares.stream()  
    .filter(AGE.greaterThan(8))
    .mapToInt(Hare::getAge)
    .sorted()
    .count();

在上面的代码中,当流到达 count() 操作时,它将检查它自己的管道。首先会推断上面例子中的 AGE 操作,其次在不改变 count() 结果的情况下,会推断 mapToInt() 和 sorted() 操作,这些操作可以被消除,因此这段代码的操作被节省为 “select count(*) from hare where age > 8”。这意味着您可以使用 Java 8 流而你不必如此在意流是如何转换为SQL的。

如何下载和加入我们
如果你想学习如何使用 Speedment 的 API 和在项目中如何使用 Speedment,可以访问网址www.speedment.org,并可以在gitter上发表评论,也可以从 GitHub 上下载 Speedment 的源码,来贡献你自己的代码。

总结
回顾早期的一些老项目,一个超过100行代码的数据库类,现在可以使用 Java 8 缩减成1行代码。那是反转后的摩尔定律,在14年内(=7摩尔周期),行数大约减半了七次。这就是进步!
  • 大小: 31.7 KB
  • 大小: 12.6 KB
来自: OneAPM
5
0
评论 共 2 条 请登录后发表评论
2 楼 kaki 2016-01-15 11:13
业务逻辑分割复杂代码可能会带来效率问题。
1 楼 yybing110 2016-01-13 18:48
世界变得太快 我有点跟不上了 ,问个弱弱的问题  如果sql很复杂的表达式 他能胜任吗 ,还有其他的级联查询等等 

发表评论

您还没有登录,请您登录后再发表评论

相关推荐

  • C#_string_byte数组转换解析

    C# string byte数组转换解析 C# string byte数组转换实现的过程是什么呢?C# string byte数组间的转换需要注意什么呢?C# string byte数组间转换所涉及的方法是什么呢?让我们来看看具体的内容: C# string byte数组转换之string类型转成byte[]: byte[] byteArray = System.Text.Encoding.Default.GetBytes ( str ); 反过来,byte[]转成string: string str = System.Text.Encoding.Default.GetString ( byteArray ); 其它编码方式的,如System.Text.UTF8Encoding,System.Text.UnicodeEncoding class等;例如: string类型转成ASCII byte[]:("01" 转成 byte[] = new byte[]{ 0x30, 0x31}) 1 byte[] byteArray = System.Text.Encoding.ASCII.GetBytes ( str ); ASCII byte[] 转成string:(byte[] = new byte[]{ 0x30, 0x31} 转成 "01") 2 string str = System.Text.Encoding.ASCII.GetString ( byteArray ); 有时候还有这样一些需求: byte[] 转成原16进制格式的string,例如0xae00cf, 转换成 "ae00cf";new byte[]{ 0x30, 0x31}转成"3031":

  • js 字符串转字节数组,字节数组转字符串(互转)

    js字符串转字节数组,js字节数组转字符串

  • js byte[] 和string 相互转换 UTF-8

    function stringToByte(str) { var bytes = new Array(); var len, c; len = str.length; for(var i = 0; i < len; i++) { c = str.charCodeAt(i); if(c >= 0x010000 && c <= 0x10FFFF) {

  • day015: JS数据类型之问——转换篇

    day015: JS数据类型之问——转换篇1. [] == ![]结果是什么?为什么?解析:== 中,左右两边都需要转换为数字然后进行比较。[]转换为数字为0。![] 首...

  • JS笔记之第一天

    JavaScript:简称JS JS的原名不是JavaScript,而是LiveScript JS分为三个部分 1.ECMAScript 标准→JS的基本的语法 2.DOM——Document Object Model 文档对象模型 3.DOM——Browser Object Model 浏览器对象模型 JavaScript是什么? 是一门脚本语言:不需要编译(翻译),直接运行 是一门解释性的语言:遇到一行代码就解释(执行)一行代码 是一门动态类型的语言 是一门基于对象的语言 是一门弱类...

  • byte[] 转化为 string 转化为汉字和字母

    public byte[] poetry1= new byte[5000]; 如果只是字母就 //ASCIIEncoding encoding = new ASCIIEncoding (); //string msg = encoding.GetString (poetry1); //string msg = Encoding.ASCII.G...

  • byte数组与字符串之间相互转换

    byte数组与字符串之间相互转换字符串转数组数组转字符串使用String类利用16进制字符串 字符串转数组 public static void main(String[] args) { String str="失败是成功之母!"; byte[] bytes = str.getBytes(StandardCharsets.UTF_8); } 数组转字符串 使用String类 public static void main(String[] args) thr

  • 最全js方法,String字符串base64编码和types字节数组相互转换

    做一个项目,用到加密,字节网上找的资料整理在这里,String字符串base64编码和types字节数组相互转换。废话不多说代码贴上.

  • JavaScript utf8 bytes转string超简单办法(支持所有文字)

    [高手勿喷] 最近想做一个Minecraft的nbt编辑器,里面使用bytes存储了字符串。但是JavaScript似乎没有bytes转字符串的功能(至少我没找到),网上查了很多方法都太繁琐了,这里提供一个又偷懒又好用的办法。 //声明一个Uint8Array,经常被用来存放字节 let bytes=new Uint8Array(....); //声明空字符串 let str=""; for(le...

  • js byte[] 和string 相互转换 UTF-8

    from:http://blog.csdn.net/u013022210/article/details/55101558 js byte[] 和string 相互转换 UTF-8 //字符串转字节序列 function stringToByte(str) { var bytes = new Array(); var len, c; len = str.le...

  • 几种字符串到byte[] 数组转化为String 的方法

    将Byte数组转化为String问题 FCL得很多方法的返回值都是包含字符的Byte数组而不是返回一个String,这样的方法包含在如下的类中: · System.Net.Sockets.Socket.Receive · System.Net.Sockets.Socket.ReceiveFrom · System.Net.Sockets.Socket.BeginReceive · Syst

  • js 字节数组转字符串

    String.fromCharCode.apply(null, new Uint8Array(data .value))

  • JavaScript把16进制数组转换为字符串数组

    单个16进制数要转换为对应的字符,可以shiyong

  • Java-String

    任何字符串都是String类的对象,字符串是不变的,它们的值在创建后无法更改 在字符串的内部,是用一串字符char[]来存储的。因为数组一经确定长度无法更改,所以字符串一经创建就无法更改。 如果两个字符串内容相同,则他俩使用同一个内存地址,所以可以共享它们 public class Demo2 { public static void main(String[] args) { String s1 = "123"; String s2 = "123"; St

  • Java开发中巧妙实现将byte[]数组转化为String类型

    很多人在Java编程时,总是喜欢用一下方法将数组转为字符串:(a为byte数组)   String s=a.toString(); 可是每次返回的时候,新手看来返回的结果是乱码,比如说我,写RSA算法时,没有注意,就以为是解密出来的乱码(哈哈哈),但其实[B@1b6d3586 为栈地址值,这个时候要知道对于返回一个String对象,new一个是基本上不会错的,测试代码如下:1 S...

  • C# string类型和byte[]类型相互转换

    string类型转成byte[]:byte[] byteArray = System.Text.Encoding.Default.GetBytes ( str );byte[]转成string:string str = System.Text.Encoding.Default.GetString ( byteArray );string类型转成ASCII byte[]:(”01” 转成 byte[]

  • day016: 谈谈你对闭包的理解

    day016: 谈谈你对闭包的理解什么是闭包?红宝书(p178)上对于闭包的定义:闭包是指有权访问另外一个函数作用域中的变量的函数, MDN 对闭包的定义为:闭包是指那...

Global site tag (gtag.js) - Google Analytics