假设我有一份文件
{
id: 1,
fruits: []
}这里的水果就像一组
现在,我想原子地为带有主键=1的文档添加一个值,或者创建这样的文档,如果它不存在(例如,使用SetInsert ReQL )
我还需要对增量执行相同的操作(ReQL .Add)
显然,这不能在客户端代码中完成,因为它破坏了原子性,并且我最终得到了不一致的数据。
我希望像这样的事情是可能的
r.table('results').insert({
id: '62c70132-6516-4279-9803-8daf407ffa5c',
counter: r.row('counter').add(1).default(0)
}, {conflict: "update"})但是它与"RqlCompileError: r.row未在此上下文中定义“一起消失。
非常感谢您的帮助和指导,谢谢!
发布于 2015-07-26 18:09:38
目前,insert不可能做到这一点。另一个提到的解决方案不是原子的,因为它使用子查询。我们正在https://github.com/rethinkdb/rethinkdb/issues/3753中为这个问题制定一个解决方案。
但是,您可以使用replace执行原子插入:
r.table('results').get('62c70132-6516-4279-9803-8daf407ffa5c')
.replace({
id: '62c70132-6516-4279-9803-8daf407ffa5c',
counter: r.row('counter').add(1).default(0)
})如果文档不存在,replace实际上将执行插入操作。
发布于 2015-07-26 07:39:34
在RethinkDB中,所有单一查询更新都是原子化的。如果require不认为某个特定的更新/替换操作是原子操作,它将引发一个错误,并要求您向查询中添加一个非原子标志。所以通常情况下,你不必担心太多。但是,这只适用于update和replace查询。使用insert是不可能以原子的方式做到这一点的。
正确的是,如果检索该文档,更新它的客户端,然后将它放回DB中,它本质上也是一个非原子的更新。
但是,在单个查询中,您可以执行以下操作,这实际上是一个重复插入,其方式与使用insert的方式相同,但使用了replace
r.table('FruitStore')
.get(1)
.replace({
id : 1,
fruits : r.row('fruits').default([]).append('orange')
})...which是原子的。类似地,要使用add操作:
r.table('FruitStore')
.get(1)
.replace({
id : 1,
count : r.row('count').default(0).add(1)
})https://stackoverflow.com/questions/31634640
复制相似问题