Spring Data Mongo + 延迟加载 + REST Jackson

Posted

技术标签:

【中文标题】Spring Data Mongo + 延迟加载 + REST Jackson【英文标题】:Spring Data Mongo + Lazy Load + REST Jackson 【发布时间】:2017-09-14 18:17:51 【问题描述】:

您好,我对带有延迟加载的 Spring 和 Mongo 有一些问题。

我有这个配置:

<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>1.5.1.RELEASE</version>
</parent>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-mongodb</artifactId>
</dependency>

本文档:

@Document
public class User 
    @Id
    private String id;

    @DBRef
    private Place place;

    @DBRef(lazy=true)
    private Country country;

    .
    .
    .

一切正常,但是当我在 RestController 中公开“用户”时,例如:

@RestController
public class UserController 

    .
    .
    .

    @RequestMapping(value = "user/idUser", method = RequestMethod.GET, produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
    public User getById(@PathVariable("idUser") String idUser)    
            return userService.getById(idUser);    
    

输出是:

  
    "id": "58ebf11ee68f2751f33ae603",
    "place": 
      "id": "58e3bf76e76877586435f5af",
      "name": "Place X"
    ,
    "country": 
      "id": "58daa782e96139070bbc851c",
      "name": "México",
      "target":
        "id": "58daa782e96139070bbc851c",
        "name": "México",
      
    
  

问题:

    如果“country”标记为“lazy=true”,为什么会打印出来?

    为什么“国家”中有一个名为“目标”的新字段?

    如何避免序列化标记为“lazy=true”的字段?

提前感谢您的帮助。

【问题讨论】:

我认为 spring-data 对 mongo 的延迟加载与 Jackson 配合得不好。 Jackson 可能使用反射,而 spring-data 延迟加载使用 CGLIB 代理对象,这可能会破坏 Jackson 在 Spring(Jackson 的)序列化到 json 中的任何反射操作。 你解决过这个问题吗? @DavidA 不,我没有,但我改变了返回数据的方式。我认为正确的方法是返回一个 DTO 而不是直接暴露实体或 mongo 文档。 我遇到了 json-patch 的目标类未知的错误。我没有办法,只能选择lazy=false @Merch0 你找到问题 1 的答案了吗? 【参考方案1】:

我在序列化结果中显示“目标”时遇到了类似的问题,并且能够通过创建自定义序列化程序来解决此问题,以便序列化实际对象,而不是具有“目标”的代理字段并且有一个时髦的类名。

显然,您可以选择不获取目标,而不是简单地获取它并对其进行序列化,如下所示:

public class DBRefSerializer extends JsonSerializer<Object> 

    @Override
    public void serialize(Object value, JsonGenerator generator, SerializerProvider provider)
            throws JsonGenerationException, IOException 

        provider.defaultSerializeValue(value, generator);
    

    @Override
    public void serializeWithType(Object value, JsonGenerator generator, SerializerProvider provider,
            TypeSerializer typeSer)
            throws IOException 

        Object target = value;
        if (value instanceof LazyLoadingProxy) 
            LazyLoadingProxy proxy = (LazyLoadingProxy)value;
            target = proxy.getTarget();
            provider.defaultSerializeValue(target, generator);
         else 
            provider.defaultSerializeValue(target, generator);
        
    

然后像这样注释你想要通过它运行的 DBRef:

@JsonSerialize(using=DBRefSerializer.class)
@DBRef(lazy = true)
private SomeClass someProperty;

显然这并不适合所有人,但我想我会发布它以防它帮助其他人。

【讨论】:

在列表上试过这个,但没有用 @JsonSerialize(using=DBRefSerializer.class) @DBRef(lazy=true) private List repaymentDataList = new ArrayList();有什么建议吗?【参考方案2】:

添加@JsonIgnore 注解。

@JsonIgnore
@DBRef(lazy = true)
private SomeClass someProperty;

来源: Cooperation MongoDB lazy loading with Jackson @JsonIgnore in SpringBoot Rest Controller

【讨论】:

我不知道您的理解是什么,您为什么认为提供的解决方案是错误的。不要认为你的看法是 OP 的看法。

以上是关于Spring Data Mongo + 延迟加载 + REST Jackson的主要内容,如果未能解决你的问题,请参考以下文章

Spring Data JPA - 在没有 @Transactional 的情况下获取延迟加载的集合

从控制台应用程序使用带有休眠功能的spring-data-jpa时如何延迟加载收集

将 mongo 查询转换为 spring-data-mongo 查询

延迟加载有效,但不应该

使用 spring data mongo 插入 Mongo 文档

如何使用 mongo 搜索集合并返回子文档列表(Spring-data-mongo)