2

在将 JPA 2.1 与 Hibernate 4.3.11 实现一起使用的 Java 应用程序中,我试图将SqlResultSetMapping本机查询结果映射到实体。查询包括两个具有相同列名的表,因此我需要使用别名并映射它们(此处描述的问题:http ://www.tinesoft.com/java/be-aware-of-mutliple-result-mappings-in- jpa )

为了简单起见,我将查询和实体缩小到最低限度,这仍然会导致问题。真实代码使用两个实体和 DB 函数,这就是使用原生查询而不是 JPQL 的原因。

网关实体:

@Entity
public class Gateway implements Serializable {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Integer id;
    private Integer active;
    ...
    @Column(name = "activation_code")
    private String activationCode;
    ...
}

SqlResultSetMapping:

@SqlResultSetMapping(
        name = "GatewayWithLoc",
        entities = {
            @EntityResult( entityClass = Gateway.class , fields =     @FieldResult(name = "id", column = "gw_id"))
        }
)

询问:

Query query = em.createNativeQuery("SELECT gw.id AS gw_id, ..., active, activation_code, ... FROM gateway gw", "GatewayWithLoc");
List<Object[]> rows = query.getResultList();

例外:

Caused by: org.postgresql.util.PSQLException: The column name activati2_1_0_ was not found in this ResultSet.
at org.postgresql.jdbc2.AbstractJdbc2ResultSet.findColumn(AbstractJdbc2ResultSet.java:2803)
...

如果我SELECT *不使用fields = @FieldResult...in SqlResultSetMapping,代码会按预期工作,但我不能这样做,因为不同表中的列名相同。

我将通过手动将结果映射到实体来解决这个问题,因为我需要快速解决方案,但是如果我做错了什么或者 Hibernate 不支持我正在尝试做的事情,那就太好了。书“Pro JPA 2”包含应该可以工作的非常相似的示例。

更新:使用 Hibernate 5.1.1 (Spring 4.3.2) 测试 - 结果相同

更新:看起来所有列名都需要按照多个答案中的说明进行指定。对我来说,这似乎是 JPA/Hibernate 的一个大问题——这些列名可以从实体派生,只有手动指定的例外。相反,我需要写 3 次列名(1. 查询,2. 结果集映射 3. 实体),这很难看且难以维护。

4

3 回答 3

0

看起来本机查询未映射到SqlResultSetMapping. 查询中的列比映射中的列多得多。查询中的选定列必须与您的映射相匹配。

于 2016-08-31T08:02:29.883 回答
0

您可以使用SELECT NEW

  1. 创建一个类具有查询中需要使用的所有列。

    package dz.immo;
    
    public class GatewayGatewayWithLoc {
        private Integer idGw;
        private Integer activeGw;
        private Integer idGwWl;
        private Integer activeGwWl;
    
            public GatewayGatewayWithLoc(Integer idGw, Integer activeGw, Integer idGwWl, Integer activeGwWl) {
                this.idGw = idGw;
                this.activeGw = activeGw;
                this.idGwWl = idGwWl;
                this.activeGwWl = activeGwWl;
            }
           ....
         // getter and setter
         ...... 
    
        }
    
  2. 使用查询

    Query query = em.createQuery("SELECT NEW dz.immo.GatewayGatewayWithLoc(gw.id, gw.active, gwwl.id, gwwl.active) FROM Gateway gw ,GatewayGatewayWithLoc gwwl WHERE ...");
    List<GatewayGatewayWithLoc> liste = query.getResultList();
    // display the answer
    for (GatewayGatewayWithLoc gwgw : liste) {
        System.out.println("GatewayGatewayWithLoc : " + gwgw.getIdGw() + " " + gwgw.getActiveGw()+ " " + gwgw.getIdGwWl()+ " " + gwgw.getActiveGwWl());
    }`
    
于 2016-08-31T15:03:15.727 回答
0

我认为您需要@FieldResult明确指定所有内容,特别是因为您的列名在两个表之间存在冲突。

于 2016-09-01T11:26:00.997 回答