首页 > 解决方案 > 调用页面大小大于 36 的 Spring Data MongoDB 存储库方法时出现 StackOverflowError

问题描述

调用 MongoDB 存储库接口的方法时出现 StackOverflowError:

public interface TermRepository extends MongoRepository<Term, String>, QuerydslPredicateExecutor<Term> {
    // [...]
    @Query("{$or:[{'apis' : {$in : ?0 } }, {$text:{$search:?1}}]}")
    Page<Term> globalSearch(List<DBRef> apis, String searchKeyword, Pageable pageable);
}
{
    $or: [{
        'apis': {
            $in: [{
                '$ref': 'api',
                '$id': ObjectId('5e3ad9faaafa595898b6a682')
            }]
        }
    }, {
        $text: {
            $search: 'account'
        }
    }]
}

有谁知道发生了什么?

术语:

public class Term {

    @Id
    private String id;

    @NotBlank
    @TextIndexed
    @JsonProperty
    private String name;

    @NotBlank
    @TextIndexed
    @JsonProperty
    private String objectType;

    @Transient
    @JsonProperty
    private String snakeCase;

    @NotEmpty
    @JsonProperty
    private List<String> functionalCategories;

    @NotNull
    @JsonProperty
    private TermTypeEnum termType;

    @NotEmpty
    @JsonProperty
    private Map<String, String> description;

    @Setter
    @NotNull
    @JsonProperty
    private TermStateEnum state;

    @NotBlank
    @TextIndexed
    @JsonProperty
    private String example;

    @DBRef
    @NotNull
    @Indexed
    @JsonProperty
    private List<Api> apis;

    @DBRef
    @NotNull
    @JsonProperty
    private User contributor;

    @NotBlank
    @TextIndexed
    @JsonProperty
    private String version;

    @DBRef
    @JsonProperty
    private Map<String, Term> attributes;
}

堆栈跟踪:https ://pastebin.com/y0XYt7p6

尽管可能存在循环引用,因为 Term 具有属性 Map<String, Term>,但我认为导致堆栈溢出的不是循环引用。

做一些调试,我看到堆栈溢出发生了,因为最后 spring-data 只实例化这个 Term (不是其他的,所以我没有看到循环引用):

Term(id=5e3ad9faaafa595898b6a7ea, name=debitCurrency, objectType=string, snakeCase=debit_currency, functionalCategories=null, termType=BODY, description={"english"=Debit Currency.}, state=PROPOSED, example=null, apis=[Api(id=5e3ad9faaafa595898b6a67f, name=Payments-1.0.1, description=null, responsible=null)], contributor=null, version=null, attributes=null)

而这又不引用其他术语。

这是另一个术语,仅供比较:

Term(id=5e3ad9faaafa595898b6a6c8, name=displayCardNumber, objectType=string, snakeCase=display_card_number, functionalCategories=null, termType=BODY, description={"english"=Related card number to the account.}, state=PROPOSED, example=null, apis=[Api(id=5e3ad9faaafa595898b6a682, name=Accounts-1.0.2, description=null, responsible=null)], contributor=null, version=null, attributes=null)

我看不出有什么区别,但是debitCurrency会产生堆栈溢出而displayCardNumber不会。

标签: javaspringspring-bootstack-overflowspring-data-mongodb

解决方案


我确实有一个循环引用(更准确地说是自我引用)。

我想出了一个解决方法,包括制作@DBRef属性lazy=true和实现类的自定义序列化程序,它Term具有自引用(或循环引用)。实现自定义序列化程序的原因是我可以保存一组已经序列化的对象,如果我必须再次序列化一个对象,我就不用了。


推荐阅读