Java Hibernate createSQLQuery using addEntity

2022-09-04 22:04:46

我需要应用与此类似的 SQL 查询。

SELECT
    id as id,
    c03 as c03,
    c34 as c34 
FROM
    (SELECT
        id,
        c03,
        c34 
    FROM
        students
    where
        c34 in(
            ?, ?,?,?
        ) 
    order by
        id desc) o 
group by
    c34;

还有我的Java代码。

private final void retrieveStudents(){
    final List result = currentSession()
            .createSQLQuery("SELECT id as id,c03 as c03,c34 as c34 FROM (SELECT id,c34,c03 FROM students where c34 in(:filters) order by id desc) o group by c34;")
            .setParameterList("filters",Arrays.asList(74,1812))
            .list();
    result.forEach(this::consumer1);
}

返回一个 objets 数组,我可以迭代,但我想返回一个 Student 对象,所以我添加。

   .addEntity(Student.class)

但是抛出一个错误,它说

Column 'ACTIVE' not found.

我也尝试过使用.addScalar,但同样的事情发生了。

.addScalar("id",org.hibernate.type.IntegerType.INSTANCE)
.addScalar("c03",org.hibernate.type.DateType.INSTANCE)

这是来自学生的一列,但不在投影上,我的问题我怎么能这样做,我只是认为以某种方式应用别名会Hibernate填充学生实体。

我想要的只是一个填充了id,c03,c34值的学生对象。

我做错了什么,这可能吗?


答案 1

对于此用例,您不希望休眠将学生视为实体,而是将其视为 DTO。

为此,不要使用该方法,而是使用:addEntitysetResultTransfomer

final List result = currentSession()
        .createSQLQuery("SELECT id as id,c03 as c03,c34 as c34 " + 
                         "FROM (SELECT id,c34,c03 FROM students " + 
                               "where c34 in(:filters) " + 
                               "order by id desc) o group by c34")
        .setParameterList("filters",Arrays.asList(74,1812))
        .addScalar("id",org.hibernate.type.IntegerType.INSTANCE)
        .addScalar("c03",org.hibernate.type.DateType.INSTANCE)
        .addScalar("c34",org.hibernate.type.DateType.INSTANCE)
        .setResultTransformer(Transformers.aliasToBean(Student.class))
        .list();

这适用于非实体类,前提是该类具有与投影列的名称匹配的 setter,并且存在一个 no-arg 构造函数。我从未在实体类上测试过它。

如果您没有 setter,但有这 3 个字段的构造函数,则可以使用 :

// choose here the right constructor
java.lang.reflect.Constructor constructor = Student.class.getConstructors()...
// ...
.setResultTransformer(new AliasToBeanConstructorResultTransformer(constructor));

相反。

编辑:我不会使用实体作为DTO(但为此用例创建一个特定的DTO):如果您的一个服务认为实体是以常规方式加载的(因此已完全初始化),对其进行一些修改(例如更新字段)并保存实体,该怎么办?

在最好的情况下,一个不可为空的字段(在数据库端)将不会被初始化,你会得到一个 ConstraintViolationException 并触发回滚,从而确保数据安全。

在最坏的情况下,您将通过设置为实体的所有字段(但三个加载的字段除外)来损坏数据。null


答案 2

推荐