java - 调用页面大小大于 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);
}
apis
是一个只有一个 DBRef 的列表: new DBRef("api", new ObjectId("5e3ad9faaafa595898b6a682"))searchKeyword
等于“帐户”pageable
是Page request [number: 0, size 37, sort: UNSORTED]
。如果size
是 36,它不会抛出 StackOverflowError!- 查询被翻译成
{
$or: [{
'apis': {
$in: [{
'$ref': 'api',
'$id': ObjectId('5e3ad9faaafa595898b6a682')
}]
}
}, {
$text: {
$search: 'account'
}
}]
}
- 如果我直接在 mongo 中执行查询,它会返回 55 个元素。
- 我试图将线程堆栈增加到 -Xss1G(我知道很多),它只是缓慢地填充堆栈并且不会返回。如果我以 36 的页面大小重新运行测试,它会立即返回。
有谁知道发生了什么?
术语:
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
不会。
解决方案
我确实有一个循环引用(更准确地说是自我引用)。
我想出了一个解决方法,包括制作@DBRef
属性lazy=true
和实现类的自定义序列化程序,它Term
具有自引用(或循环引用)。实现自定义序列化程序的原因是我可以保存一组已经序列化的对象,如果我必须再次序列化一个对象,我就不用了。
推荐阅读
- java - 使用硬连线到代码中的用户名和密码在 Spring Security 中进行身份验证的过程
- c# - 如何为地图坐标实现 IEnumerator?
- java - notifyItemChanged(int position) 仅更新 Recyclerview 中的单个项目
- javascript - CSS 窗帘,网站运行时会打开
- macos - Pyusb 和 Libusb 在 MacOS 上提供 NoBackendError
- python - 未找到超参数优化 Python 中的 Leaky ReLU
- c - C alloc / free struct arr
- c# - C#方法或对象不能被第二次调用
- javascript - 使用水平滚动处理溢出内容
- python - 我想使用类和数据结构创建一个简单的库存管理系统。我是 python 中的 OOP 新手