JDO 2.0查询语言的特点

  查询语言的改进是JDO2.0规范中的重要环节,本文从较高的层面阐述JDO2.0所提供的一些新功能。由于JDO2.0规范还未进入公开草案状态,目前还没有任何内容敲定下来,一切都还可能面临变化。不过,JDO2.0将会很快进入最后阶段,而这里提到的查询特性是JDO2.0专家组(译者注:David Jordan就是专家组重要成员)花费时间最多,并且相对来说最为稳定。因此,我有足够理由相信,最终规范与这里的描述将会基本一致。

  如果各位读者觉得本文遗漏了某些重要的特性,建议立即到JDO论坛(http://www.jdocentral.com/forums/index.php?showforum=10)去提出并讨论。这里我们需要感谢JDO2.0规范领导人Craig Russell授权给我公开这些JDO2.0查询语言的新特性。

  查询结果

  你可以创建一个对A类的查询,通过contains()引用到B类,再通过又一层contains()引用到C类,最后再使用一个“.”操作符引用到D类。但最终返回的集合中只会包含A类的对象实例,如果要从结果中获得其它类,就必须通过A类的引用来逐个获取相关的其它类对象。如果你的查询条件里面包含了B、C或D类的约束,那么在结果集中通过A类对象引用其它类对象时,必须重新将这些约束在Java代码中重复一遍,也就是说,你不得不在Java和JDOQL中重复声明限制条件。再者,你可能只关心满足查询条件的D类对象,而不希望中间的B、C类对象被JDO底层创建从而节省内存或相关资源。

  在JDO2.0中,你再也不受缚于这些限制了。你可以返回:

  数据类(PersistentCapable)的一个或多个字段
  候选类以外的其它类对象

  统计数据

  这意味着你可以返回A、B、C、D类对象,或者它们的某些字段,或者二者的混和结果。你还可以计算类似min或max之类的统计结果。基本上,你想返回什么结果都可以。

  当你创建一个查询时,你可以指定一个“结果定义(Result Specification)”来指定返回什么样的内容。它是一个包含一个或多个以逗号分隔的“结果表达式(Result Expression)”。结果表达式可以是:

  this关键字,表示返回候选类的对象实例。这与JDO1.0是一样的字段,标明候选类或引用类的某个字段的值,如 address.street.name 字段表达式,代表对多个字段进行JDO预定义的几种算术运算而获得的结果变量,代表查询条件中出现的某个中间变量引用表达式,也就是JDO1.0中的通过“.”操作符进行的对象之间的引用 统计表达式通过对以上这几种结果表达式的组合运用,你可以获得任何你想要的结果。

  JDO2.0支持下面的统计函数:

  count(表达式),表达式可以是this
  sum(数字型字段表达式),“数字型字段表达式”可以是通过字段或字段的运算得到的数字型的结果
  min(数字型字段表达式)
  max(数字型字段表达式)
  avg(数字型字段表达式)

  对查询结果的指定是通过下面的API:

  void javax.jdo.Query.setResult(String result)

  如果你不调用这个方法,或者参数是null,则返回候选类的对象实例(相当于设置为“this”),即JDO1.0的返回结果。如果你只指定了唯一结果表达式,则返回集合的元素类型与该结果的类型一致。另外,在默认方式下,如果指定了多个结果表达式,则返回的集合元素类型将是 Object[]。

  你可以在结果定义字符串的开头标上distinct来保证结果不会重复。而如果结果定义串中包含好几个表达式,那么distinct可以保证结果集中不会有重复的数据组。

  每个结果表达式可以指定一个名称,对于简单的字段,系统会默认以该字段的名称作为结果中该项的名称。对复杂的表达式,你可以使用下面的语法指定名称:

  result_expression as name
  
  名称的使用可以让结果中的该项作为结果类中的一个属性来进行设置和使用。你可以指定一个结果类(result class),用来返回查询结果。如果查询结果是一个单值,结果类可以是任何JDO支持的类(Integer, Double, String, BigInteger, BigDecimal, java.util.Date, java.sql.Date, java.sql.Time, java.sql.Timestamp)。Query中设置结果类的方法是:

  void setResultClass(Class resultClass)

  如果查询结果包含多个结果表达式,你可以定义一个结果类来保留结果中的各项数据,这个类必须有一个无参数的构造器。此外,每个结果表达式必须对应此类中的一个属性,不论是一个public的字段,还是一个public的setXxx()方法,并且这种直接的或bean风格的属性名称与查询结果中各项结果表达式的名称保持一致。

    组操作(Grouping)

  统计功能可用于一个分组操作中。JDO2.0提供类似SQL中的GROUP BY和HAVING子句。Query方法:

  void setGrouping(String groupSpec)

  用于指定分组的原则。groupSpec参数包含一个或多个以逗号分隔的分组表达式,还可以跟上一个以“having ”开头的过滤条件。调用此方法后,setResult()参数的每个结果表达式项必须是groupSpec中的一项,或者是groupSpec中的一项或多项的运算结果。所有groupSpec项的值均相同的结果被归在同一组中(同一条结果记录)。having子句的过滤条件可以包含boolean结果的判断语句或者是对分组表达式的统计运算。与SQL一样,having子句用于对分组后的结果集进行条件过滤。

  唯一性(Uniqueness)

  很多人一直奇怪为什么Query的执行(execute)结果是一个Object类型的对象,因为这样给开发人员造成必须手工强制将结果转换成Collection的不便。JDO1.0中查询结果一般有多个元素,但JDO专家组计划在JDO2.0中加入对返回单值结果的查询的支持,于是将Query的执行结果定义为Object类型。

  你有时会执行一个你确定结果只会有一条的查询(比如统计总数或者按具有唯一索引的会员帐号查找对象等等),在JDO2.0中,你可以调用Query方法来声明:

  void setUnique(boolean unique)

  当你传入“true”后,Query执行的结果将是一个单独的值对象,如果无任何结果返回,则结果是null。如果JDO发现查询结果返回了多条记录,则会扔出一个异常。

  限制返回结果的大小

  设计用户界面的时候,我们常常会显示结果集的某一部分子集(比如分页显示或只显示前十条之类)。为了性能和效率,你多半会需要限制返回结果的范围。Query的方法可以完成这一点:

  void setRange(int fromInclusive, int toExclusive)

  该方法返回的结果集只包含原结果集的第fromInclusive条到第toExclusive-1条。

  新的过滤条件操作符

  一些新的操作符被加到JDOQL中,以便执行针对引用、Map、字符串和数字的操作。instanceof操作符返回一个boolean值,可以让你过滤某个指定类的对象;同样返回boolean值的containsKey(Object)函数和containsValue(Object)用于访问Map元素。

  字符串处理方面加入了很多函数,toLowerCase()和toUpperCase()分别完成大小写转换,另外还有下列函数用于查找子串位置和获得子串:

  int indexOf(String)
  int indexOf(String, int) String substring(int) String substring(int,int)

  另外,String的方法:

  boolean matches(String pattern)

  用于执行正则式匹配。目前只能提供有限的匹配功能。“.”和“.*”可表示通配符,而“(?i)”表示匹配不区分大小写。

  对数字型的字段,JDO2.0增加了两个函数:

  Math.abs(numeric) Math.sqrt(numeric)

  预定义查询(Named Queries)

  你可以在JDO描述符(metadata)中声明常用的JDOQL查询语句,这样就不用将查询嵌入到Java源代码中。这样可以提供一些灵活性,比如将查询语句写到一个配置好的文本文件中,而需要修改时可以直接改该文件,而不用更改Java源码。描述符中的每个查询都有一个名字,而要执行某个查询时,可以使用下面的方法来创建查询:

  Query newNamedQuery(Class cls, String queryName)

  JDO将会搜索描述符来找到对应的预定义查询语句并生成相应的Query对象。

  访问静态字段(static fields)

  你将可以在JDOQL中访问数据类中以public static final方式声明的常量。比如

  public static final int FEMALE = 0;
  public static final int MALE = 1;
  public static final int UNKNOWN = 2;

  而在查询中使用类似“salary > 5000.0 && gender == MALE”的过滤条件。

  批量删除(Deletion by Query)

  在JDO1.0中,要删除一个对象,必须先将其载入内存,再删除,然而很多情况下,你在删除之前并不需要访问该对象,这样的做法比较低效。在JDO2.0中,提供了Query的几个方法来删除符合查询条件的一组对象:

  Object deletePersistentAll(Object[] parameters) Object deletePersistentAll(Map parameters) Object deletePersistentAll()

  此查询结果的对象会被全部从数据库删除。这些方法返回被删除对象的集合。你的程序可以决定是否需要对被删除的对象逐个访问。如果你不访问这些对象,Query执行的性能将不会受到任何影响,换句话说,这些对象将不会在内存中生成。
  
  厂商扩展的查询特性

  JDO厂商可以给JDOQL提供各种各样的特殊查询功能。每个厂商可以定义一组属于该厂商自己的扩展功能。如果你需要使用其中的功能,你需要将该厂商的扩展加到程序运行环境中。每个扩展有一个名字和一个可选的值。你可以单独设置每个扩展,或者一次性设置多个扩展。Query的下列方法用于设置厂商扩展特性:

  void setExtensions(Map extensions) void addExtension(String key, Object value)

  SQL直接访问

  如果JDO2.0的查询功能不能满足需要,并且以下条件都满足:

  你的应用运行在关系数据库上
  
  你需要的查询有SQL语句可以实现

  你知道从你的类模型到数据库的映射细节

  你就可以创建一个SQL查询,方法是调用PersistentManger的方法:

  Query newQuery(String language, Object query)
  
  每一个参数需要设置为“javax.jdo.query.SQL”,而query是一个具体的SQL语句。如果你的查询需要返回类的实例,则这个SQL语句必须返回该类的相应主键字段。使用直接的SQL时,JDOQL提供的函数将不能被使用,否则将会扔出异常JDOUserException,比如,你不能对该Query设置过滤条件字符串、排序声明或变量声明。使用SQL查询时,参数都是未指定类型的,在SQL语句中以“?”表示,并且按出现的顺序被绑定。

  结束语

  各位读者已经看到,JDO2.0对查询语句增加了很多新的功能和特性,我认为其中的查询结果定义是最大的改进。正如我在文章开头说的,这一部分变化目前来说是JDO2.0中最稳定的部分,不会有大的变化。厂商可以开始提供很多这样的功能,不用考虑专家组还在讨论的JDO2.0其余部分将如何变化。

  在JDO2.0提供标准的对象/关系映射的前提下,这里描述的查询功能和一个用于多层结构的JDO应用的脱钩/挂钩(detach/attach)机制,将使JDO占领更多的市场。对这一点有深切体会的厂商将不遗余力地争取率先推出稳定、完整、高效的JDO2.0产品。JDO将是基于数据库存储和管理的应用开发的首先API,开发人员们将享受面向对象的模型设计和高效的数据存储管理系统。

(0)

相关推荐

  • Java数据对象JDO 2.0查询语言的特点

    查询语言的改进是JDO2.0规范中的重要环节,本文从较高的层面阐述JDO2.0所提供的一些新功能.由于JDO2.0规范还未进入公开草案状态,目前还没有任何内容敲定下来,一切都还可能面临变化.不过,JDO2.0将会很快进入最后阶段,而这里提到的查询特性是JDO2.0专家组(译者注:David Jordan就是专家组重要成员)花费时间最多,并且相对来说最为稳定.因此,我有足够理由相信,最终规范与这里的描述将会基本一致. 如果各位读者觉得本文遗漏了某些重要的特性,建议立即到JDO论坛(http://w

  • JDO 2.0查询语言的特点

    查询语言的改进是JDO2.0规范中的重要环节,本文从较高的层面阐述JDO2.0所提供的一些新功能.由于JDO2.0规范还未进入公开草案状态,目前还没有任何内容敲定下来,一切都还可能面临变化.不过,JDO2.0将会很快进入最后阶段,而这里提到的查询特性是JDO2.0专家组(译者注:David Jordan就是专家组重要成员)花费时间最多,并且相对来说最为稳定.因此,我有足够理由相信,最终规范与这里的描述将会基本一致. 如果各位读者觉得本文遗漏了某些重要的特性,建议立即到JDO论坛(http://w

  • 基于Spring中各个jar包的作用及依赖(详解)

    先附spring各版本jar包下载链接http://repo.spring.io/release/org/springframework/spring/ spring.jar 是包含有完整发布模块的单个jar 包.但是不包括mock.jar, aspects.jar, spring-portlet.jar, and spring-hibernate2.jar 示例图片为Spring-2.5.6.jar的包目录 下面讲解各个jar包的作用: 1.org.springframework.aop或sp

  • JavaBean(EJB) 3.0 全新体验

    引言 期待以久的EJB3.0规范在最近发布了它的初稿.在本文中将对新的规范进行一个概要性的介绍,包括新增的元数据支持,EJBQL的修改,实体Bean模型访问bean上下文的新方法和运行时环境等等.作者还讨论了EJB在未来要作出的调整以及EJB3.0与其他开发规范之间的关系. 开始 无论如何由于EJB的复杂性使之在J2EE架构中的表现一直不是很好.EJB大概是J2EE架构中唯一一个没有兑现其能够简单开发并提高生产力的组建.EJB3.0规范正尝试在这方面作出努力以减轻其开发的复杂性.EJB3.0减轻

  • thinkphp查询,3.X 5.0方法(亲试可行)

    一.介绍 ThinkPHP内置了非常灵活的查询方法,可以快速的进行数据查询操作,查询条件可以用于读取.更新和删除等操作,主要涉及到where方法等连贯操作即可,无论是采用什么数据库,你几乎采用一样的查询方法(个别数据库例如Mongo在表达式查询方面会有所差异),系统帮你解决了不同数据库的差异性,因此我们把框架的这一查询方式称之为查询语言.查询语言也是ThinkPHP框架的ORM亮点,让查询操作更加简单易懂.下面来一一讲解查询语言的内涵. 二.查询方式 ThinkPHP可以支持直接使用字符串作为查

  • ThinkPHP3.1查询语言详解

    ThinkPHP的查询语言配合连贯操作可以很好解决复杂的业务逻辑需求,本篇我们就首先来深入了解下框架的查询语言. 1.查询语言介绍 ThinkPHP内置了非常灵活的查询方法,可以快速的进行数据查询操作,查询条件可以用于读取.更新和删除等操作,主要涉及到where方法等连贯操作即可,无论是采用什么数据库,你几乎采用一样的查询方法(个别数据库例如Mongo在表达式查询方面会有所差异),系统帮你解决了不同数据库的差异性,因此我们把框架的这一查询方式称之为查询语言.查询语言也是ThinkPHP框架的OR

  • 简单了解Spring Framework5.0新特性

    SpringFramework5.0是自2013年12月版本4发布之后SpringFramework的第一个主发行版.SpringFramework项目的领导人JuergenHoeller于2016年7月28日宣布了第一个SpringFramework5.0里程碑版本(5.0M1). 现在,将近一年的时间过去以后,我们期盼已久的RC3版本将于2017年7月18日发行.这是路线图规划中SpringFramework5.0首个GA发行版的最后一次发行. 从高层来看,SpringFramework5.

  • linux安装mysql 8.0.x的完整步骤

    MySQL MySQL简介 MySQL原本是一个开放源代码的关系数据库管理系统,原开发者为瑞典的MySQL AB公司,该公司于2008年被昇阳微系统(Sun Microsystems)收购.2009年,甲骨文公司(Oracle)收购昇阳微系统公司,MySQL成为Oracle旗下产品. MySQL在过去由于性能高.成本低.可靠性好,已经成为最流行的开源数据库,因此被广泛地应用在Internet上的中小型网站中.随着MySQL的不断成熟,它也逐渐用于更多大规模网站和应用,比如维基百科.Google和

  • Centos7下安装MySQL8.0.23的步骤(小白入门级别)

    首先简单介绍一下什么叫MySQL: 数据库简而言之就是存储数据的仓库,为了方便数据的存储和管理,它将数据按照特定的规律存储在磁盘上.是为了实现一定的目的,按照某种规则组织起来的数据的集合: MySQL是一种开放源代码的关系型数据库管理系统(RDBMS),使用最常用的数据库管理语言--结构化bai查询语言(SQL)进行数据库管理.通过数据库管理系统,可以有效的组织和管理存储在数据库中的数据. 那么我们常见的数据库模型: 1,关系型数据库 MySql Oracle SQL Server 2,非关系型

  • 详解ASP.NET Core 2.0 路由引擎之网址生成(译)

    问题 如何在ASP.NET Core 2.0中由路由引擎来生成网址? 答案 新建一个空项目,修改Startup.cs文件,添加MVC服务和中间件: public void ConfigureServices(IServiceCollection services) { services.AddMvc(); } public void Configure(IApplicationBuilder app, IHostingEnvironment env) { if (env.IsDevelopmen

随机推荐