Jooq DSL一对多关系

3

我正在使用Spring Data + Jooq DSL。 作为结果实体,我正在使用非Jooq生成的简单实体,不带任何注释,并且对于一对多关系获取结果:

[{
   "id":2,
   "name":"James",
   "addresses":[
      {
         "id":null,
         "country":null,
         "street":null
      }
   ]
}]

有没有一种方法返回一个空数组来表示地址?

我的代码用于执行请求:

    public Set<User> getUserById(Set<Long> id) {
        Set<User> result = new HashSet<>();
        ResultQuery users = dsl.select(
                field("u.id", Long.class).as("id"),
                field("u.name", String.class).as("name"),
                field("a.id", Long.class).as("addresses_id"),
                field("a.country", String.class).as("addresses_country"),
                field("a.street", String.class).as("addresses_street")
        ).from("schema.user_table u")
                .leftJoin("schema.address_table a")
                .on("u.id = a.user_id")
                .where(field("u.id").in(id));
        try(ResultSet rs = users.fetchResultSet()) {
            JdbcMapper<User> mapper = JdbcMapperFactory
                    .newInstance()
                    .addKeys("id")
                    .newMapper(User.class);
            result = mapper.stream(rs).collect(Collectors.toSet());
        } catch (Exception e) {
            e.printStackTrace();
        }
        return result;
    }

1
请查看此链接 https://blog.jooq.org/2021/07/06/jooq-3-15s-new-multiset-operator-will-change-how-you-think-about-sql/ - Simon Martinelli
谢谢,它可以工作,但我认为它太复杂了,因为当前的spring-boot版本提供了jooq 3.14.13,而multiset是3.15的新功能。我已经将我的修复添加为答案。 - Roman Kiselev
您始终可以在Spring Boot中使用最新版本的jOOQ。只需在属性中设置jooq.version即可。为什么不使用jOOQ生成器呢? - Simon Martinelli
2个回答

1

为什么不直接使用SQL/JSON从数据库中直接生成JSON文档呢?

public String getUserById(Set<Long> id) {
    return dsl.select(coalesce(jsonArrayAgg(
        jsonObject(
            key("id").value(field("u.id", Long.class)),
            key("name").value(field("u.name", String.class)),
            key("addresses").value(coalesce(
                jsonArrayAgg(jsonObject(
                    key("id").value(field("a.id", Long.class)),
                    key("country").value(field("a.country", String.class)),
                    key("street").value(field("a.street", String.class))
                )),
                jsonArray()
            ))
        ),
        jsonArray()
    )))
    .from("schema.user_table u")
    .leftJoin("schema.address_table a")
    .on("u.id = a.user_id")
    .where(field("u.id").in(id))
    .fetchSingle().value1().data();
}

如果您确实需要中间的User表示,那么您可以选择以下方式之一:
public Set<User> getUserById(Set<Long> id) {
    return dsl.select(
        field("u.id", Long.class),
        field("u.name", String.class),
        multisetAgg(
            field("a.id", Long.class),
            field("a.country", String.class),
            field("a.street", String.class)
        ).convertFrom(r -> r == null 
            ? Collections.<Address>emptyList() 
            : r.map(Records.mapping(Address::new)))
    )
    .from("schema.user_table u")
    .leftJoin("schema.address_table a")
    .on("u.id = a.user_id")
    .where(field("u.id").in(id))
    .fetchSet(Records.mapping(User::new));
}

关于代码生成和执行的附注

虽然与本问题没有直接关联,但如果您的模式是静态的(在编译时不知道),我真的建议您重新考虑使用源代码生成。如果您不使用它,您将错过很多jOOQ API的优势,就像在执行jOOQ查询时使用其他工具一样。


谢谢您的回答和建议,我已经开始使用生成的实体进行工作。 - Roman Kiselev

0

对我来说,将addressId指定为关键字起作用:

.addKeys("id", "addresses_id")

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接