`
hai0378
  • 浏览: 532002 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

PA 2.0 中的动态类型安全查询

 
阅读更多

JPA 2.0 中的动态类型安全查询

如何通过 Criteria API 构建动态查询并减少运行时失败

如果编译器能够对查询执行语法正确性检查,那么对于 Java 对象而言该查询就是类型安全的。Java™Persistence API (JPA) 的 2.0 版本引入了 Criteria API,这个 API 首次将类型安全查询引入到 Java 应用程序中,并为在运行时动态地构造查询提供一种机制。本文介绍如何使用 Criteria API 和与之密切相关的 Metamodel API 编写动态的类型安全查询。

Pinaki Poddar, 高级软件工程师, IBM

2009 年 11 月 09 日

  • +内容

自从 JPA 于 2006 年首次被引入之后,它就得到了 Java 开发社区的广泛支持。该规范的下一个主要更新 —— 2.0 版本 (JSR 317) —— 将在 2009 年年底完成(见 参考资料)。JPA 2.0 引入的关键特性之一就是 Criteria API,它为 Java 语言带来了一种独特的能力:开发一种 Java 编译器可以在运行时验证其正确性的查询。Criteria API 还提供一个能够在运行时动态地构建查询的机制。

本文将介绍 Criteria API 和与之密切相关的 元模型(metamodel)概念。您将学习如何使用 Criteria API 开发 Java 编译器能够检查其正确性的查询,从而减少运行时错误,这种查询优于传统的基于字符串的 Java Persistence Query Language (JPQL) 查询。借助使用数据库函数或匹配模板实例的样例查询,我将演示编程式查询构造机制的强大威力,并将其与使用预定义语法的 JPQL 查询进行对比。本文假设您具备基础的 Java 语言编程知识,并了解常见的 JPA 使用,比如 EntityManagerFactory或 EntityManager

JPQL 查询有什么缺陷?

JPA 1.0 引进了 JPQL,这是一种强大的查询语言,它在很大程度上导致了 JPA 的流行。不过,基于字符串并使用有限语法的 JPQL 存在一些限制。要理解 JPQL 的主要限制之一,请查看清单 1 中的简单代码片段,它通过执行 JPQL 查询选择年龄大于 20 岁的 Person列表:

清单 1. 一个简单(并且错误)的 JPQL 查询
EntityManagerem = ...; 
 String jpql = "select p from Person where p.age > 20"; 
 Queryquery = em.createQuery(jpql); 
 List result = query.getResultList();

这个基础的例子显示了 JPA 1.0 中的查询执行模型的以下关键方面:

  • JPQL 查询被指定为一个 String(第 2 行)。
  • EntityManager是构造一个包含给定 JPQL 字符串的 可执行查询实例的工厂(第 3 行)。
  • 查询执行的结果包含无类型的 java.util.List的元素。

但是这个简单的例子有一个验证的错误。该代码能够顺利通过编译,但将在运行时 失败,因为该 JPQL 查询字符串的语法有误。清单 1的第 2 行的正确语法为:

 String jpql = "select p from Person p where p.age > 20";

不幸的是,Java 编译器不能发现此类错误。在运行时,该错误将出现在第 3 或第 4 行(具体行数取决于 JPA 提供者是否在查询构造或执行期间根据 JPQL 语法解析 JPQL 字符串)。

类型安全查询如何提供帮助?

Criteria API 的最大优势之一就是禁止构造语法错误的查询。清单 2 使用 CriteriaQuery接口重新编写了 清单 1中的 JPQL 查询:

清单 2. 编写 CriteriaQuery的基本步骤
EntityManagerem = ... 
 QueryBuilderqb = em.getQueryBuilder(); 
 CriteriaQuery<Person> c = qb.createQuery(Person.class); 
 Root<Person> p = c.from(Person.class); 
 Predicatecondition = qb.gt(p.get(Person_.age), 20); 
 c.where(condition); 
 TypedQuery<Person> q = em.createQuery(c); 
 List<Person> result = q.getResultList();

清单 2展示了 Criteria API 的核心构造及其基本使用:

  • 第 1 行通过几种可用方法之一获取一个 EntityManager实例。
  • 在第 2 行,EntityManager创建 QueryBuilder的一个实例。QueryBuilder是 CriteriaQuery的工厂。
  • 在第 3 行,QueryBuilder工厂构造一个 CriteriaQuery实例。CriteriaQuery被赋予泛型类型。泛型参数声明 CriteriaQuery在执行时返回的结果的类型。在构造 CriteriaQuery时,您可以提供各种结果类型参数 —— 从持久化实体(比如 Person.class)到形式更加灵活的Object[]
  • 第 4 行在 CriteriaQuery实例上设置了 查询表达式。查询表达式是在一个树中组装的核心单元或节点,用于指定 CriteriaQuery。图 1 显示了在 Criteria API 中定义的查询表达式的层次结构:

    图 1. 查询表达式中的接口层次结构
    查询表达式中的接口层次结构

    首先,将 CriteriaQuery设置为 Person.class查询。结果返回 Root<Person>实例 pRoot是一个查询表达式,它表示持久化实体的范围。Root<T>实际上表示:“对所有类型为 T的实例计算这个查询。” 这类似于 JPQL 或 SQL 查询的 FROM子句。另外还需要注意,Root<Person>是泛型的(实际上每个表达式都是泛型的)。类型参数就是表达式要计算的值的类型。因此 Root<Person>表示一个对Person.class进行计算的表达式。

  • 第 5 行构造一个 PredicatePredicate是计算结果为 true 或 false 的常见查询表达式形式。谓词由 QueryBuilder构造,QueryBuilder不仅是 CriteriaQuery的工厂,同时也是查询表达式的工厂。QueryBuilder包含构造传统 JPQL 语法支持的所有查询表达式的 API 方法,并且还包含额外的方法。在 清单 2中,QueryBuilder用于构造一个表达式,它将计算第一个表达式参数的值是否大于第二个参数的值。方法签名为:

     Predicate gt(Expression<? extends Number> x, Number y);

    这个方法签名是展示使用强类型语言(比如 Java)定义能够检查正确性并阻止错误的 API 的好例子。该方法签名指定,仅能将值为 Number的表达式与另一个值也为 Number的表达式进行比较(例如,不能与值为 String的表达式进行比较):

     Predicate condition = qb.gt(p.get(Person_.age), 20);

    第 5 行有更多学问。注意 qb.gt()方法的第一个输入参数:p.get(Person_.age),其中 p是先前获得的 Root<Person>表达式。p.get(Person_.age)是一个 路径表达式。路径表达式是通过一个或多个持久化属性从根表达式进行导航得到的结果。因此,表达式p.get(Person_.age)表示使用 Person的 age属性从根表达式 p导航。您可能不明白 Person_.age是什么。您可以将其暂时看作一种表示Person的 age属性的方法。我将在谈论 JPA 2.0 引入的新 Metamodel API 时详细解释 Person_.age

    如前所述,每个查询表达式都是泛型的,以表示表达式计算的值的类型。如果 Person.class中的 age属性被声明为类型 Integer(或int),则表达式 p.get(Person_.age)的计算结果的类型为 Integer。由于 API 中的类型安全继承,编辑器本身将对无意义的比较抛出错误,比如:

     Predicate condition = qb.gt(p.get(Person_.age, "xyz"));
  • 第 6 行在 CriteriaQuery上将谓词设置为其 WHERE子句。
  • 在第 7 行中,EntityManager创建一个可执行查询,其输入为 CriteriaQuery。这类似于构造一个输入为 JPQL 字符串的可执行查询。但是由于输入 CriteriaQuery包含更多的类型信息,所以得到的结果是 TypedQuery,它是熟悉的 javax.persistence.Query的一个扩展。如其名所示,TypedQuery知道执行它返回的结果的类型。它是这样定义的:

     public interface TypedQuery<T> extends Query { 
                 List<T> getResultList(); 
     }

    与对应的无类型超接口相反:

     public interface Query { 
     List getResultList(); 
     }

    很明显,TypedQuery结果具有相同的 Person.class类型,该类型在构造输入 CriteriaQuery时由 QueryBuilder指定(第 3 行)。

  • 在第 8 行中,当最终执行查询以获得结果列表时,携带的类型信息展示了其优势。得到的结果是带有类型的 Person列表,从而使开发人员在遍历生成的元素时省去麻烦的强制类型转换(同时减少了 ClassCastException运行时错误)。

现在归纳 清单 2中的简单例子的基本方面:

  • CriteriaQuery是一个查询表达式节点树。在传统的基于字符串的查询语言中,这些表达式节点用于指定查询子句,比如 FROMWHEREORDER BY。图 2 显示了与查询相关的子句:
    图 2. CriteriaQuery封装了传统查询的子句
    查询表达式的接口层次结构
  • 查询表达式被赋予泛型。一些典型的表达式是:
    • Root<T>,相当于一个 FROM子句。
    • Predicate,其计算为布尔值 true 或 false(事实上,它被声明为 interface Predicate extends Expression<Boolean>)。
    • Path<T>,表示从 Root<?>表达式导航到的持久化属性。Root<T>是一个没有父类的特殊 Path<T>
  • QueryBuilder是 CriteriaQuery和各种查询表达式的工厂。
  • CriteriaQuery被传递给一个可执行查询并保留类型信息,这样可以直接访问选择列表的元素,而不需要任何运行时强制类型转换。
 

持久化域的元模型

讨论 清单 2时指出了一个不常见的构造:Person_.age,它表示 Person的持久化属性 age清单 2使用 Person_.age形成一个路径表达式,它通过 p.get(Person_.age)从 Root<Person>表达式 p导航而来。Person_.age是 Person_类中的公共静态字段,Person_是 静态、已实例化的规范元模型类,对应于原来的 Person实体类。

元模型类描述持久化类的元数据。如果一个类安装 JPA 2.0 规范精确地描述持久化实体的元数据,那么该元模型类就是 规范的。规范的元模型类是 静态的,因此它的所有成员变量都被声明为 静态的(也是 public的)。Person_.age是静态成员变量之一。您可以在开发时在源代码中生成一个具体的 Person_.java来 实例化一个规范类。实例化之后,它就可以在编译期间以强类型的方式引用 Person的持久化属性。

这个 Person_metamodel 类是引用 Person的元信息的一种代替方法。这种方法类似于经常使用(有人可能认为是滥用)的 Java Reflection API,但概念上有很大的不同。您可以使用反射获得关于 java.lang.Class的实例的元信息,但是不能以编译器能够检查的方式引用关于Person.class的元信息。例如,使用反射时,您将这样引用 Person.class中的 age字段:

 Field field = Person.class.getField("age");

不过,这种方法也存在很大的限制,类似于 清单 1中基于字符串的 JPQL 查询存在的限制。编译器能够顺利编译该代码,但不能确定它是否可以正常工作。如果该代码包含任何错误输入,它在运行时肯定会失败。反射不能实现 JPA 2.0 的类型安全查询 API 要实现的功能。

类型安全查询 API 必须让您的代码能够引用 Person类中的持久化属性 age,同时让编译器能够在编译期间检查错误。JPA 2.0 提供的解决办法通过静态地公开相同的持久化属性实例化名为 Person_的元模型类(对应于 Person)。

关于元信息的讨论通常都是令人昏昏欲睡的。所以我将为熟悉的 Plain Old Java Object (POJO) 实体类展示一个具体的元模型类例子(domain.Person),如清单 3 所示:

清单 3. 一个简单的持久化实体
packagedomain; 
 @Entity 
 public classPerson { 
  @Id 
  private longssn; 
  privatestring name; 
  private intage; 

  // public gettter/setter methods 
  publicString getName() {...} 
 }

这是 POJO 的典型定义,并且包含注释(比如 @Entity或 @Id),从而让 JPA 提供者能够将这个类的实例作为持久化实体管理。

清单 4 显示了 domain.Person的对应静态规范元模型类:

清单 4. 一个简单实体的规范元模型
packagedomain; 
 importjavax.persistence.metamodel.SingularAttribute; 

 @javax.persistence.metamodel.StaticMetamodel(domain.Person.class) 

 public class Person_{ 
  public static volatile SingularAttribute<Person,Long> ssn; 
  public static volatile SingularAttribute<Person,String> name; 
  public static volatile SingularAttribute<Person,Integer> age; 
 }

元模型类将原来的 domain.Person实体的每个持久化属性声明为类型为 SingularAttribute<Person,?>的静态公共字段。通过利用这个Person_元模型类,可以在编译期间引用 domain.Person的持久化属性 age—不是通过 Reflection API,而是直接引用静态的 Person_.age字段。然后,编译器可以根据 age属性声明的类型实施类型检查。我已经列举了一个关于此类限制的例子:QueryBuilder.gt(p.get(Person_.age), "xyz")将导致编译器错误,因为编译器通过 QueryBuilder.gt(..)的签名和 Person_.age的类型可以确定 Person的 age属性是一个数字字段,不能与 String进行比较。

其他一些需要注意的要点包括:

  • 元模型 Person_.age字段被声明为类型 javax.persistence.metamodel.SingularAttributeSingularAttribute是 JPA Metamodel API 中定义的接口之一,我将在下一小节描述它。SingularAttribute<Person, Integer>的泛型参数表示该类声明原来的持久化属性和持久化属性本身的类型。
  • 元模型类被注释为 @StaticMetamodel(domain.Person.class)以将其标记为一个与原来的持久化 domain.Person实体对应的元模型类。

Metamodel API

我将一个元模型类定义为一个持久化实体类的描述。就像 Reflection API 需要其他接口(比如 java.lang.reflect.Fieldjava.lang.reflect.Method)来描述 java.lang.Class的组成一样,JPA Metamodel API 也需要其他接口(比如 SingularAttributePluralAttribute)来描述元模型类的类型及其属性。

图 3 显示了在 Metamodel API 中定义用于描述类型的接口:

图 3. Metamodel API 中的持久化类型的接口的层次结构
图 3. Metamodel API 中的持久化类型的接口的层次结构

图 4 显示了在 Metamodel API 中定义用于描述属性的接口:

图 4. Metamodel API 中的持久化属性的接口的层次结构
图 4. Metamodel API 中的持久化属性的接口的层次结构

JPA 的 Metamodel API 接口比 Java Reflection API 更加专业化。需要更细微的差别来表达关于持久化的丰富元信息。例如,Java Reflection API 将所有 Java 类型表示为 java.lang.Class。即没有通过独立的定义对概念进行区分,比如类、抽象类和接口。当然,您可以询问 Class它是一个接口还是一个抽象类,但这与通过两个独立的定义表示接口和抽象类的差别不同。

Java Reflection API 在 Java 语言诞生时就被引入(对于一种常见的多用途编程语言而言,这曾经是一个非常前沿的概念),但是经过多年的发展才认识到强类型系统的用途和强大之处。JPA Metamodel API 将强类型引入到持久化实体中。例如,持久化实体在语义上区分为MappedSuperClassEntity和 Embeddable。在 JPA 2.0 之前,这种语义区分是通过持久化类定义中的对应类级别注释来表示的。JPA Metamodel 在 javax.persistence.metamodel包中描述了 3 个独立的接口(MappedSuperclassTypeEntityType和 EmbeddableType),以更加鲜明的对比它们的语义特征。类似地,可以通过接口(比如 SingularAttributeCollectionAttribute和 MapAttribute)在类型定义级别上区分持久化属性。

除了方便描述之外,这些专门化的元模型接口还有实用优势,能够帮助构建类型安全的查询从而减少运行时错误。您在前面的例子中看到了一部分优势,随着我通过 CriteriaQuery描述关于连接的例子,您将看到更多优势。

运行时作用域

一般而言,可以将 Java Reflection API 的传统接口与专门用于描述持久化元数据的 javax.persistence.metamodel的接口进行比较。要进一步进行类比,则需要对元模型接口使用等效的运行时作用域概念。java.lang.Class实例的作用域由 java.lang.ClassLoader在运行时划分。一组相互引用的 Java 类实例必须在 ClassLoader作用域下定义。作用域的边界是 严格或 封闭的,如果在 ClassLoader L作用域下定义的类 A试图引用不在 ClassLoader L作用域之内的类 B,结果将收到可怕的 ClassNotFoundException或 NoClassDef FoundError(对于处理包含多个ClassLoader的环境的开发人员或部署人员而言,问题就复杂了)。

现在将一组严格的可相互引用的类称为运行时作用域,而在 JPA 1.0 中称为 持久化单元。持久化单元作用域的持久化实体在 META-INF/persistence.xml 文件的 <class>子句中枚举。在 JPA 2.0 中,通过 javax.persistence.metamodel.Metamodel接口让开发人员可以在运行时使用作用域。Metamodel接口是特定持久化单元知道的所有持久化实体的容器,如图 5 所示:

图 5. 元模型接口是持久化单元中的类型的容器
图 5. 元模型接口是持久化单元中的类型的容器

这个接口允许通过元模型元素的对应持久化实体类访问元模型元素。例如,要获得对 Person持久化实体的持久化元数据的引用,可以编写:

 EntityManagerFactory emf = ...; 
 Metamodel metamodel = emf.getMetamodel(); 
 EntityType<Person> pClass = metamodel.entity(Person.class);

这是一个用类的名称通过 ClassLoader获得 Class的类比:

 ClassLoader classloader =  Thread.currentThread().getContextClassLoader(); 
 Class<?> clazz = classloader.loadClass("domain.Person");

可以在运行时浏览 EntityType<Person>获得在 Person实体中声明的持久化属性。如果应用程序在 pClass(比如pClass.getSingularAttribute("age", Integer.class))上调用一个方法,它将返回一个 SingularAttribute<Person, Integer>实例,该实例与实例化规范元模型类的静态 Person_.age成员相同。最重要的是,对于应用程序可以通过 Metamodel API 在运行时引用的属性,是通过实例化静态规范元模型 Person_类向 Java 编译器提供的。

除了将持久化实体分解为对应的元模型元素之外,Metamodel API 还允许访问所有已知的元模型类 (Metamodel.getManagedTypes()),或者通过类的持久化信息访问元模型类,例如 embeddable(Address.class),它将返回一个 EmbeddableType<Address>实例(ManagedType<>的子接口)。

在 JPA 中,关于 POJO 的元信息使用带有源代码注释(或 XML 描述符)的持久化元信息进一步进行区分 —— 比如类是否是嵌入的,或者哪个字段用作主键。持久化元信息分为两大类:持久化(比如 @Entity)和映射(比如 @Table)。在 JPA 2.0 中,元模型仅为持久化注释(不是映射注释)捕捉元数据。因此,使用当前版本的 Metamodel API 可以知道哪些字段是持久化的,但不能找到它们映射到的数据库列。

规范和非规范

尽管 JPA 2.0 规范规定了规范的静态元模型类的精确样式(包括元模型类的完整限定名及其静态字段的名称),应用程序也能够编写这些元模型类。如果应用程序开发人员编写元模型类,这些类就称为 非规范元模型。现在,关于非规范元模型的规范还不是很详细,因此对非规范元模型的支持不能在 JPA 提供者之间移植。您可能已经注意到,公共静态字段仅在规范元模型中 声明,而没有初始化。声明之后就可以在开发CriteriaQuery时引用这些字段。但是,必须在运行时给它们赋值才有意义。尽管为规范元模型的字段赋值是 JPA 提供者的责任,但非规范元模型则不存在这一要求。使用非规范元模型的应用程序必须依赖于特定供应商机制,或开发自己的机制来在运行时初始化元模型属性的字段值。

代码生成和可用性

自动生成源代码通常让人担心。为规范元模型生成源代码更是如此。生成的类在开发期间使用,构建CriteriaQuery的其他代码部分直接在运行时引用它们,从而留下一些可用性问题:

  • 生成的源代码文件应该在初始源代码所在的目录中,还是在一个独立的或与输出目录相关的目录中?
  • 是否应该在版本控制配置管理系统中检查源代码文件?
  • 如何维护初始 Person实体定义及其规范 Person_元模型之间的对应关系?例如,如果要编辑 Person.java 以添加其他持久化属性,或通过重构重命名持久化属性,那么应该怎么办呢?

在撰写本文时,这些问题还没有确切的答案。

注释处理和元模型生成

如果您有许多持久化实体,您将倾向于不亲自编写元模型类,这是很自然的事情。持久化提供者 应该为您生成这些元模型类。在规范中没有强制规定这种工具或生成机制,但是 JPA 之间已经私下达成共识,他们将使用在 Java 6 编译器中集成的 Annotation Processor 工具生成规范元模型。Apache OpenJPA 提供一个工具来生成这些元模型类,其生成方式有两种,一是在您为持久化实体编译源代码时隐式地生成,二是通过显式地调用脚本生成。在 Java 6 以前,有一个被广泛使用的称为 apt的 Annotation Processor 工具,但在 Java 6 中,编译器和 Annotation Processor 的合并被定义为标准的一部分。

要像持久化提供者一样在 OpenJPA 中生成这些元模型类,仅需在编译器的类路径中使用 OpenJPA 类库编译 POJO 实体:

 $ javac domain/Person.java

将生成规范元模型 Person_类,它将位于 Person.java 所在的目录,并且作为该编译的一部分。

 

编写类型安全的查询

到目前为止,我已经构建了 CriteriaQuery的组件和相关的元模型类。现在,我将展示如何使用 Criteria API 开发一些查询。

函数表达式

函数表达式将一个函数应用到一个或多个输入参数以创建新的表达式。函数表达式的类型取决于函数的性质及其参数的类型。输入参数本身可以是表达式或文本值。编译器的类型检查规则与 API 签名结合确定什么是合法输入。

考虑一个对输入表达式应用平均值的单参数表达式。CriteriaQuery选择所有 Account的平均余额,如清单 5 所示:

清单 5. CriteriaQuery中的函数表达式
CriteriaQuery<Double> c = cb.createQuery(Double.class); 
 Root<Account> a = c.from(Account.class); 

 c.select(cb.avg(a.get(Account_.balance)));

等效的 JPQL 查询为:

 String jpql = "select avg(a.balance) from Account a";

在 清单 5中,QueryBuilder工厂(由变量 cb表示)创建一个 avg()表达式,并将其用于查询的 select()子句。

Fluent API

如这个例子所示,Criteria API 方法经常返回可以直接在相关方法中使用的类型,从而提供了一种称为 Fluent API 的流行编程风格。

该查询表达式是一个构建块,可以通过组装它为查询定义最后的选择谓词。清单 6中的例子显示了通过导航到 Account的余额创建的 Path表达式,然后 Path表达式被用作两个二进制函数表达式(greaterThan()和 lessThan())的输入表达式,这两个表达式的结果都是一个布尔表达式或一个谓词。然后,通过 and()操作合并谓词以形成最终的选择谓词,查询的where()子句将计算该谓词:

清单 6. CriteriaQuery中的 where()谓词
CriteriaQuery<Account> c = cb.createQuery(Account.class); 
 Root<Account> account = c.from(Account.class); 
 Path<Integer> balance = account.get(Account_.balance); 
 c.where(cb.and 
       (cb.greaterThan(balance, 100), 
        cb.lessThan(balance), 200)));

等效的 JPQL 查询为:

"select a from Account a where a.balance>100 and a.balance<200";

符合谓词

某些表达式(比如 in())可以应用到多个表达式。清单 7 给出了一个例子:

清单 7. CriteriaQuery中的多值表达式
CriteriaQuery<Account> c = cb.createQuery(Account.class); 
 Root<Account> account = c.from(Account.class); 
 Path<Person> owner = account.get(Account_.owner); 
 Path<String> name = owner.get(Person_.name); 
 c.where(cb.in(name).value("X").value("Y").value("Z"));

这个例子通过两个步骤从 Account进行导航,创建一个表示帐户所有者的名称的路径。然后,它创建一个使用路径表达式作为输入的 in()表达式。in()表达式计算它的输入表达式是否等于它的参数之一。这些参数通过 value()方法在 In<T>表达式上指定,In<T>的签名如下所示:

 In<T> value(T value);

注意如何使用 Java 泛型指定仅对值的类型为 T的成员计算 In<T>表达式。因为表示 Account所有者的名称的路径表达式的类型为 String,所以与值为 String类型的参数进行比较才有效,String值参数可以是字面量或计算结果为 String的另一个表达式。

将 清单 7中的查询与等效(正确)的 JPQL 进行比较:

"select a from Account a where a.owner.name in ('X','Y','Z')";

在 JPQL 中的轻微疏忽不仅不会被编辑器检查到,它还可能导致意外结果。例如:

"select a from Account a where a.owner.name in (X, Y, Z)";

连接关系

尽管 清单 6和 清单 7中的例子将表达式用作构建块,查询都是基于一个实体及其属性之上的。但是查询通常涉及到多个实体,这就要求您将多个实体 连接起来。CriteriaQuery通过 类型连接表达式连接两个实体。类型连接表达式有两个类型参数:连接源的类型和连接目标属性的可绑定类型。例如,如果您想查询有一个或多个 PurchaseOrder没有发出的 Customer,则需要通过一个表达式将 Customer连接到PurchaseOrder,其中 Customer有一个名为 orders类型为 java.util.Set<PurchaseOrder>的持久化属性,如清单 8 所示:

清单 8. 连接多值属性
CriteriaQuery<Customer> q = cb.createQuery(Customer.class); 
 Root<Customer> c = q.from(Customer.class); 
 SetJoin<Customer, PurchaseOrder> o = c.join(Customer_.orders);

连接表达式从根表达式 c创建,持久化属性 Customer.orders由连接源(Customer)和 Customer.orders属性的可绑定类型进行参数化,可绑定类型是 PurchaseOrder而 不是已声明的类型 java.util.Set<PurchaseOrder>。此外还要注意,因为初始属性的类型为 java.util.Set,所以生成的连接表达式为 SetJoin,它是专门针对类型被声明为 java.util.Set的属性的 Join。类似地,对于其他受支持的多值持久化属性类型,该 API 定义 CollectionJoinListJoin和 MapJoin。(图 1显示了各种连接表达式)。在 清单 8的第 3 行不需要进行显式的转换,因为CriteriaQuery和 Metamodel API 通过覆盖 join()的方法能够识别和区分声明为 java.util.Collection或 List或者 Set或 Map的属性类型。

在查询中使用连接在连接实体上形成一个谓词。因此,如果您想要选择有一个或多个未发送 PurchaseOrder的 Customer,可以通过状态属性从连接表达式 o进行导航,然后将其与 DELIVERED状态比较,并否定谓词:

Predicatep = cb.equal(o.get(PurchaseOrder_.status), Status.DELIVERED) 
        .negate();

创建连接表达式需要注意的一个地方是,每次连接一个表达式时,都会返回一个新的表达式,如清单 9 所示:

清单 9. 每次连接创建一个唯一的实例
SetJoin<Customer, PurchaseOrder> o1 = c.join(Customer_.orders); 
 SetJoin<Customer, PurchaseOrder> o2 = c.join(Customer_.orders); 
 asserto1 == o2;

清单 9中对两个来自相同表达式 c的连接表达式的等同性断言将失败。因此,如果查询的谓词涉及到未发送并且值大于 $200 的PurchaseOrder,那么正确的构造是将 PurchaseOrder与根 Customer表达式连接起来(仅一次),把生成的连接表达式分配给本地变量(等效于 JPQL 中的范围变量),并在构成谓词时使用本地变量。

使用参数

回顾一下本文初始的 JPQL 查询(正确那个):

 String jpql = "select p from Person p where p.age > 20";

尽管编写查询时通常包含常量文本值,但这不是一个良好实践。良好实践是参数化查询,从而仅解析或准备查询一次,然后再缓存并重用它。因此,编写查询的最好方法是使用命名参数:

 String jpql = "select p from Person p where p.age > :age";

参数化查询在查询执行之前绑定参数的值:

Queryquery = em.createQuery(jpql).setParameter("age", 20); 
 List result = query.getResultList();

在 JPQL 查询中,查询字符串中的参数以命名方式(前面带有冒号,例如 :age)或位置方式(前面带有问号,例如 ?3)编码。在CriteriaQuery中,参数本身就是查询表达式。与其他表达式一样,它们是强类型的,并且由表达式工厂(即 QueryBuilder)构造。然后,可以参数化 清单 2中的查询,如清单 10 所示:

清单 10. 在 CriteriaQuery中使用参数
ParameterExpression<Integer> age = qb.parameter(Integer.class); 
 Predicatecondition = qb.gt(p.get(Person_.age), age); 
 c.where(condition); 
 TypedQuery<Person> q = em.createQuery(c); 
 List<Person> result = q.setParameter(age, 20).getResultList();

比较该参数使用和 JPQL 中的参数使用:参数表达式被创建为带有显式类型信息 Integer,并且被直接用于将值 20绑定到可执行查询。额外的类型信息对减少运行时错误十分有用,因为阻止参数与包含不兼容类型的表达式比较,或阻止参数与不兼容类型的值绑定。JPQL 查询的参数不能提供任何编译时安全。

清单 10中的例子显示了一个直接用于绑定的未命名表达式。还可以在构造参数期间为参数分配第二个名称。对于这种情况,您可以使用这个名称将参数值绑定到查询。不过,您不可以使用位置参数。线性 JPQL 查询字符串中的整数位置有一定的意义,但是不能在概念模型为查询表达式树的 CriteriaQuery上下文中使用整数位置。

JPA 查询参数的另一个有趣方面是它们没有内部值。值绑定到可执行查询上下文中的参数。因此,可以合法地从相同的 CriteriaQuery创建两个独立可执行的查询,并为这些可执行查询的相同参数绑定两个整数值。

预测结果

您已经看到 CriteriaQuery在执行时返回的结果已经在 QueryBuilder构造 CriteriaQuery时指定。查询的结果被指定为一个或多个 预测条件。可以通过两种方式之一在 CriteriaQuery接口上指定预测条件:

CriteriaQuery<T> select(Selection<? extends T> selection); 
 CriteriaQuery<T> multiselect(Selection<?>... selections);

最简单并且最常用的预测条件是查询候选类。它可以是隐式的,如清单 11 所示:

清单 11. CriteriaQuery默认选择的候选区段
CriteriaQuery<Account> q = cb.createQuery(Account.class); 
 Root<Account> account = q.from(Account.class); 
 List<Account> accounts = em.createQuery(q).getResultList();

在 清单 11中,来自 Account的查询没有显式地指定它的选择条件,并且和显式地选择的候选类一样。清单 12 显示了一个使用显式选择条件的查询:

清单 12. 使用单个显式选择条件的 CriteriaQuery
CriteriaQuery<Account> q = cb.createQuery(Account.class); 
 Root<Account> account = q.from(Account.class); 
 q.select(account); 
 List<Account> accounts = em.createQuery(q).getResultList();

如果查询的预测结果不是候选持久化实体本身,那么可以通过其他几个构造方法来生成查询的结果。这些构造方法包含在 QueryBuilder接口中,如清单 13 所示:

清单 13. 生成查询结果的方法
 <Y> CompoundSelection<Y> construct(Class<Y> result, Selection<?>... terms); 
    CompoundSelection<Object[]> array(Selection<?>... terms); 
    CompoundSelection<Tuple> tuple(Selection<?>... terms);

清单 13中的方法构建了一个由其他几个可选择的表达式组成的预测条件。construct()方法创建给定类参数的一个实例,并使用来自输入选择条件的值调用一个构造函数。例如,如果 CustomerDetails—一个非持久化实体 —有一个接受 String和 int参数的构造方法,那么CriteriaQuery可以通过从选择的 Customer—一个持久化实体 —实例的名称和年龄创建实例,从而返回 CustomerDetails作为它的结果,如清单 14 所示:

清单 14. 通过 construct()将查询结果包放入类的实例
CriteriaQuery<CustomerDetails> q = cb.createQuery(CustomerDetails.class); 
 Root<Customer> c = q.from(Customer.class); 
 q.select(cb.construct(CustomerDetails.class, 
              c.get(Customer_.name), c.get(Customer_.age));

可以将多个预测条件合并在一起,以组成一个表示 Object[]或 Tuple的复合条件。清单 15 显示了如何将结果包装到 Object[]中:

清单 15. 将结果包装到 Object[]
CriteriaQuery<Object[]> q = cb.createQuery(Object[].class); 
 Root<Customer> c = q.from(Customer.class); 
 q.select(cb.array(c.get(Customer_.name), c.get(Customer_.age)); 
 List<Object[]> result = em.createQuery(q).getResultList();

这个查询返回一个结果列表,它的每个元素都是一个长度为 2 的 Object[],第 0 个数组元素为 Customer的名称,第 1 个数组元素为Customer的年龄。

Tuple是一个表示一行数据的 JPA 定义接口。从概念上看,Tuple是一个 TupleElement列表 —其中 TupleElement是源自单元和所有查询表达式的根。包含在 Tuple中的值可以被基于 0 的整数索引访问(类似于熟悉的 JDBC 结果),也可以被 TupleElement的别名访问,或直接通过TupleElement访问。清单 16 显示了如何将结果包装到 Tuple中:

清单 16. 将查询结果包装到 Tuple
CriteriaQuery<Tuple> q = cb.createTupleQuery(); 
 Root<Customer> c = q.from(Customer.class); 
 TupleElement<String> tname = c.get(Customer_.name).alias("name"); 
 q.select(cb.tuple(tname, c.get(Customer_.age).alias("age"); 
 List<Tuple> result = em.createQuery(q).getResultList(); 
 String name = result.get(0).get(name); 
 String age  = result.get(0).get(1);

嵌套限制

从理论上讲,可以通过嵌套 Tuple等条件(它的元素本身为 Object[]或 Tuple)来构成复杂的结果。不过,JPA 2.0 规范禁止此类嵌套。multiselect()的输入条件不能是数组或值为二元组的复合条件。允许作为multiselect()参数的唯一复合条件由 construct()方法创建(该方法仅表示一个元素)。

不过,OpenJPA 没有限制在一个复合选择条件中嵌套其他的复合选择条件。

这个查询返回一个结果列表,它的每个元素都是一个 Tuple。反过来,每个二元组都带有两个元素 —可以被每个 TupleElement的索引或别名(如果有的话)访问,或直接被TupleElement访问。清单 16中需要注意的两点是 alias()的使用,它是将一个名称绑定到查询表达式的一种方式(创建一个新的副本),和 QueryBuilder上的 createTupleQuery()方法,它仅是 createQuery(Tuple.class)的代替物。

这些能够改变结果的方法的行为和在构造期间被指定为 CriteriaQuery的类型参数结果共同组成 multiselect()方法的语义。这个方法根据最终实现结果的 CriteriaQuery的结果类型解释它的输入条件。要像 清单 14一样使用 multiselect()构造 CustomerDetails实例,您需要将 CriteriaQuery的类型指定为 CustomerDetails,然后使用将组成CustomerDetails构造方法的条件调用 multiselect(),如清单 17 所示:

清单 17. 基于结果类型的 multiselect()解释条件
CriteriaQuery<CustomerDetails> q = cb.createQuery(CustomerDetails.class); 
 Root<Customer> c = q.from(Customer.class); 
 q.multiselect(c.get(Customer_.name), c.get(Customer_.age));

因为查询结果类型为 CustomerDetailsmultiselect()将其预测条件解释为 CustomerDetails构造方法参数。如将查询指定为返回 Tuple,那么带有相同参数的 multiselect()方法将创建 Tuple实例,如清单 18 所示:

清单 18. 使用 multiselect()方法创建 Tuple实例
CriteriaQuery<Tuple> q = cb.createTupleQuery(); 
 Root<Customer> c = q.from(Customer.class); 
 q.multiselect(c.get(Customer_.name), c.get(Customer_.age));

如果以 Object作为结果类型或没有指定类型参数时,multiselect()的行为会变得更加有趣。在这些情况中,如果 multiselect()使用单个输入条件,那么返回值将为所选择的条件。但是如果 multiselect()包含多个输入条件,结果将得到一个 Object[]

 

高级特性

到目前为止,我主要强调了 Criteria API 的强类型,以及它如何帮助减少出现在基于字符串 JPQL 查询中的语义错误。Criteria API 还是以编程的方式构建查询的机制,因此通常被称为 动态查询 API。编程式查询构造 API 的威力是无穷的,但它的利用还取决于用户的创造能力。我将展示 4 个例子:

  • 使用弱类型的 API 构建动态查询
  • 使用数据库支持的函数作为查询表达式来扩展语法
  • 编辑查询实现 “在结果中搜索” 功能
  • 根据例子进行查询 —数据库社区熟悉的模式

弱类型和动态查询构建

Criteria API 的强类型检查基于开放期间的实例化元模型类的可用性。不过,在某些情况下,选择的实体仅能够在运行时决定。为了支持这种用法,Criteria API 方法提供一个并列版本,其中持久化属性通过它们的名称进行引用(类似于 Java Reflection API),而不是引用实例化静态元模型属性。该 API 的这个并列版本可以通过牺牲编译时类型检查来真正地支持动态查询构造。清单 19 使用弱类型 API 重新编写了 清单 6中的代码:

清单 19. 弱类型查询
 Class<Account> cls =Class.forName("domain.Account"); 
 Metamodelmodel = em.getMetamodel(); 
 EntityType<Account> entity = model.entity(cls);
 CriteriaQuery<Account> c = cb.createQuery(cls); 
 Root<Account> account = c.from(entity); 
 Path<Integer> balance = account.<Integer>get("balance"); 
 c.where(cb.and 
       (cb.greaterThan(balance, 100), 
        cb.lessThan(balance), 200)));

不过,弱类型 API 不能够返回正确的泛型表达式,因此生成一个编辑器来警告未检查的转换。一种消除这些烦人的警告消息的方法是使用 Java 泛型不常用的工具:参数化方法调用,比如 清单 19中通过调用 get()方法获取路径表达式。

可扩展数据库表达式

动态查询构造机制的独特优势是它的语法是可扩展的。例如,您可以在 QueryBuilder接口中使用 function()方法创建数据库支持的表达式:

 <T> Expression<T> function(String name, Class<T> type, Expression<?>...args);

function()方法创建一个带有给定名称和 0 个或多个输入表达式的表达式。function()表达式的计算结果为给定的类型。这允许应用程序创建一个计算数据库的查询。例如,MySQL 数据库支持 CURRENT_USER()函数,它为服务器用于验证当前客户机的 MySQL 帐户返回一个由用户名和主机名组成的 UTF-8 字符串。应用程序可以在 CriteriaQuery中使用未带参数的 CURRENT_USER()函数,如清单 20 所示:

清单 20. 在 CriteriaQuery中使用特定于数据库的函数
CriteriaQuery<Tuple> q = cb.createTupleQuery(); 
 Root<Customer> c = q.from(Customer.class); 
 Expression<String> currentUser = 
    cb.function("CURRENT_USER", String.class, (Expression<?>[])null); 
 q.multiselect(currentUser, c.get(Customer_.balanceOwed));

注意,在 JPQL 中不能表达等效的查询,因为它的语法仅支持固定数量的表达式。动态 API 不受固定数量表达式的严格限制。

可编辑查询

可以以编程的方式编辑 CriteriaQuery。可以改变查询的子句,比如它的选择条件、WHERE子句中的选择谓词和 ORDER BY子句中的排序条件。可以在典型的 “在结果中搜索” 工具中使用这个编辑功能,以添加更多限制在后续步骤中进一步细化查询谓词。

清单 21 中的例子创建了一个根据名称对结果进行排序的查询,然后编辑该查询以根据邮政编码进行查询:

清单 21. 编辑 CriteriaQuery
CriteriaQuery<Person> c = cb.createQuery(Person.class); 
 Root<Person> p = c.from(Person.class); 
 c.orderBy(cb.asc(p.get(Person_.name))); 
 List<Person> result = em.createQuery(c).getResultList(); 
 // start editing 
List<Order> orders = c.getOrderList(); 
 List<Order> newOrders = new ArrayList<Order>(orders); 
 newOrders.add(cb.desc(p.get(Person_.zipcode))); 
 c.orderBy(newOrders); 
 List<Person> result2 = em.createQuery(c).getResultList();

OpenJPA 中的常驻计算

通过使用 OpenJPA 的扩展特性,清单 21中的 “在结果中搜索” 例子可以在 内存中计算编辑查询,从而获得更高的效率。这个例子表明编辑查询的结果是原始结果的严格子集。因为 OpenJPA 能够在指定候选集合时从内存计算查询,所以仅需修改 清单 21的最后一行代码就可以提供原始查询的结果:

List<Person> result2 = 
em.createQuery(c).setCandidateCollection(result).getResultList();

在 CriteriaQuery上的 setter 方法 —select()where()或 orderBy()—使用新的参数替换先前的值。对应的 getter 方法(比如 getOrderList())返回的列表不是 活动的,即在返回列表上添加或删除元素不会导致修改 CriteriaQuery;另外,一些供应商甚至返回不可变的列表以阻止意外使用。因此,良好的实践是在添加和删除新的表达式之前,将返回列表复制到一个新的列表中。

根据例子进行查询

动态查询 API 中的另一个有用特性就是它能够轻松地支持 根据例子进行查询。根据例子进行查询(由 IBM®Research 在 1970 年开发出来)通常被作为早期的软件终端用户可用性例子引用。根据例子进行查询的理念使用模板实例,而不是为查询指定精确的谓词。有了给定的模板实例之后,将创建一个联合谓词,其中每个谓词都是模板实例的非 null 和非默认属性值。执行该查询将计算谓词以查找所有与模板实例匹配的实例。根据例子进行查询曾考虑添加到 JPA 2.0 中,但最终没有添加。OpenJPA 通过它的扩展 OpenJPAQueryBuilder接口支持这种查询,如清单 22 所示:

清单 22. 使用 OpenJPA 的 CriteriaQuery根据例子进行查询
CriteriaQuery<Employee> q = cb.createQuery(Employee.class); 

 Employee example = new Employee(); 
 example.setSalary(10000); 
 example.setRating(1); 

 q.where(cb.qbe(q.from(Employee.class), example);

如这个例子所示,OpenJPA 的 QueryBuilder接口扩展支持以下表达式:

 public <T> Predicateqbe(From<?, T> from, T template);

这个表达式根据给定模板实例的属性值生成一个联合谓词。例如,这个查询将查询所有薪水为 10000评级为 1的 Employee。要进一步控制比较,可以指定不用于比较的可选属性,以及为值为 String的属性指定比较方式。(参考资料部分提供一个 Javadoc 链接,可以找到关于 OpenJPA 的 CriteriaQuery扩展的说明)。

 

结束语

本文介绍了 JPA 2.0 中的新 Criteria API,它是一个用 Java 语言开发动态、类型安全的查询的机制。CriteriaQuery在运行时被构建为一个强类型查询表达式树,本文通过一系列例子展示了它的用法。

本文还确立了 Metamodel API 的关键角色,并展示了实例化元模型类如何使编译器能够检查查询的正确性,从而避免语法有误的 JPQL 查询引起的运行时错误。除了保证语法正确之外,JPA 2.0 以编程的方式构造查询的特性还能通过数据库函数实现更多强大的用途,比如通过例子进行查询。我希望本文的读者能够发现这些强大的新 API 的其他新用途。

致谢

我对 Rainer Kwesi Schweigkoffer 仔细审阅本文并提供宝贵建议表示感谢,并感谢 JPA 2.0 Expert Group 的成员深入解释了这个强大的 API 的细节。我还要感谢 Fay Wang 对本文的贡献,以及感谢 Larry Kestila 和 Jeremy Bauer 在为 OpenJPA 开发这个 Criteria API 期间提供的支持。

分享到:
评论

相关推荐

    USB2.0公头

    通过这些详细的检测报告和认证文件,我们可以了解到USB2.0公头在制造过程中的质量控制措施,确保其满足高标准的安全性和可靠性要求。 综上所述,USB2.0公头不仅在物理结构上进行了精心设计以确保稳定的数据传输和...

    TDS-PA01适配器手册

    在安装过程中,用户需要注意适配器的接口类型,防止误插导致设备损坏。 然后,手册通常会包含适配器的配置过程,这可能涉及到驱动程序的安装、设备管理器中的识别和设置调整。用户需要了解如何在操作系统中识别并...

    Z9PA-D8_Series USER GUIDE 说明书

    华硕公司对Z9PA-D8系列主板用户手册中的所有内容保留所有权利,并声明未经许可不得进行任何形式的复制、发行等行为。此外,对于用户使用手册过程中可能出现的任何问题,华硕仅在法律允许的范围内承担责任,并明确...

    HCIA-HarmonyOS Application Development2.0学习笔记

    - **动态预览**:预览器中可操作应用/服务的交互动作。 - **多端设备预览器**:支持不同设备的同时预览。 - **本地模拟器**与**远程模拟器**:分别适用于不同场景下的应用测试。 - **Remote Device 远程真机设备资源...

    《智能网联汽车技术路线图2.0》核心内容摘要.docx

    到2025年,目标是PA(部分自动化)、CA(有条件自动化)级别的智能网联汽车市场渗透率达到50%,C-V2X终端新车装配率达到50%,并且在特定场景和限定区域内实现高度自动驾驶车辆的商业化应用。2030年,这些指标将...

    服务器与数据安全.ppt.ppt

    常见的RISC处理器有Sun的Ultra SPARC V9、IBM的Power、HP的PA-RISC 2.0以及Alpha的MIPS IV。 服务器选型: 选择服务器时,需考虑应用类型、可靠性和可用性(RAS)。设备的RAS特性包括冗余部件、热交换、动态重配置...

    hp-ux反汇编

    PA-RISC 2.0架构作为HP-UX的一个重要组成部分,在HP的计算机系统中扮演着核心角色。了解其架构特点和指令集不仅有助于更好地利用HP-UX系统的性能优势,还能为开发人员和安全研究人员提供重要的工具和支持。通过掌握...

    重庆宇通 TM 5900现场电源信号输入(HART)隔离式安全栅(一入一出).pdf

    例如,若输入为变送器4~20mA(带配电),输出为4~20mA时,订货代码为TM5900-PA-V2.0。 综上所述,TM 5900现场电源信号输入(HART)隔离式安全栅是一款综合性能优秀的安全隔离设备,既具备优秀的信号处理和隔离...

    pci_dss_v2_summary_of_changes.pdf

    从1.2.1版到2.0版的过渡,不仅仅是对现有要求的小幅调整,而是对整个框架进行了全面审视和优化,旨在提高支付系统的安全性,保护持卡人的信息安全。本文将详细介绍这些变更,并深入解析其背后的意义。 #### 变更...

    K2000M系统PLC调试手册

    - 对于手册中未提及的操作,建议参考《KND用户手册》,若《KND用户手册》中也无相关信息,则视为不可操作。 #### 三、电子盘功能的重要性 - **备份操作**:完成机床调试后,应立即使用电子盘功能将系统当前数据打包...

    CNG汽车加气站操作工考试题试题库.doc

    例如,10mmHg等于1×10³Pa,而1mmHg等于1×10²Pa。这些知识对于监控和调整加气站设备的压力参数是必要的。 2. **灭火器类型与使用**:不同类型的灭火器有不同的使用注意事项。例如,二氧化碳灭火器在喷射时应防止...

    服务器基本常识介绍.ppt

    不同类型的服务器如小型机(如Sun Ultra SPARC V9、IBM Power、HP PA-RISC 2.0、Alpha MIPS IV)和PC服务器(如IBM Netfinity7100),各有优势,小型机在速度和稳定性方面更强,而PC服务器在模块化和易用性上更具...

    钢构桥建模进阶

    - **材料2**(合金钢):密度7800kg/m³,弹性模量2.0×10^11 Pa,泊松比0.3。 #### 四、定义截面 1. **桥墩截面**:采用矩形截面,顺桥向宽1m,横桥向宽3m。 2. **主梁截面**:采用自定义的箱形截面。首先,在CAD...

    HP-UX 11i internal

    此操作系统以其强大的稳定性、可靠性和安全性著称,在企业级应用环境中被广泛采用。 本书作为第一本专门针对HP-UX及HP的PA-RISC架构的技术资源书籍,不仅适用于技术支持人员、内核模块开发者以及对系统内部运作机制...

    声音传感器的应用电路

    5. **灵敏度**:-52dBV/Pa(约2.0mV/Pa),表明它对于微弱的声音也有较好的捕捉能力。 #### 四、管脚定义与注意事项 - **管脚定义**:声音传感器有两个管脚,其中一管脚与外壳相连,为负端;另一管脚为正端。 - **...

    废气方案.pdf

    9. 材料类型:例如“50/63/75mmPVC”表明了所使用的材料类型和尺寸,这对于废气处理系统中的管道、容器等部件选择至关重要,因为不同的材料有不同的耐腐蚀性、耐温性等特性。 10. 设备参数的组合:通过上面提及的...

    服务器基础介绍的PPT

    不同厂商的服务器产品各有特色,如Sun的Ultra SPARC V9、IBM的Power、HP的PA-RISC 2.0和Alpha的MIPS IV等,这些处理器的发展历程也反映了服务器技术的进步。 总之,服务器是网络基础设施中的关键部分,其性能、稳定...

    建筑施工组织2021-便桥-栈桥-平台-吊箱计算说明(9-15).doc

    在计算参数部分,钢套箱的结构行为采用弹性模型模拟,弹性模量为2.0×10^11 Pa,泊松比为0.3,质量密度为7850 kg/m³。混凝土的密度为2500 kg/m³。计算模型采用了三维有限单元法,用空间梁单元和壳单元对物理模型...

    铝合金门窗工程技术规范流程(江苏省版).doc

    规程强调了铝合金门窗在设计、生产、施工过程中的全面质量控制,以确保铝合金门窗在使用过程中的安全性、耐久性和节能效果。同时,它也为江苏省内的铝合金门窗工程提供了明确的操作指南,促进了铝合金门窗行业的规范...

Global site tag (gtag.js) - Google Analytics