- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在使用“$”运算符更新我的集合时,我遇到了这个奇怪的问题。假设以下架构:
var mySchema = {
myArray : [{
name : String
}]
};
以及以下数据(推送到 myArray 时会自动引入 _id 属性)
[{name : 'foo', _id:'1'}, {name: 'bar', _id: '2'}]
以下内容按预期工作,它替换文档而不弄乱 _id 字段:
var newFoo = {name : 'newFoo'};
mySchema.findOneAndUpdate({name : 'foo'}, {'$set' : {'myArray.0': newFoo }}},function(error, doc) {
//the resulting document looks like
//{name : 'newFoo', _id : '1'}
});
但是使用 $ 运算符而不是数组索引,将导致丢失 _id 字段:
mySchema.findOneAndUpdate({name : 'foo'}, {'$set' : {'myArray.$': newFoo }}},function(error, doc) {
//the resulting document looks like
//{name : 'newFoo'}
});
有人遇到过类似的错误吗?
谢谢。
最佳答案
这不是一个“错误”,而是设计使然,你做错了你真正想做的事情,而且这里发生的事情也不是你想象的那样。
在 MongoDB 的“更新”中,语句的“更新”部分中的任何参数都被视为“字面意思”。 $set
运算符的存在使您可以指定一个或多个“要更新的字段”,并且仅更新提到的字段而不是覆盖整个文档。
即使您在此处指定数组字段作为参数,但实际上发生的是“整个”数组元素被“替换”,而不仅仅是您的意图中的“名称”字段。
此列表演示了正在发生的情况:
var async = require('async'),
mongoose = require('mongoose'),
Schema = mongoose.Schema;
var testSchema = new Schema({
"list" : [{ "name": String }]
});
var Test = mongoose.model( 'Test', testSchema, "test" );
mongoose.connect('mongodb://localhost/test');
async.series([
function(callback) {
Test.remove({},callback);
},
function(callback) {
var test = new Test({ "list": [{ "name": "foo"},{ "name": "bar" }] });
test.save(function(err,doc) {
console.log(doc);
callback();
});
},
function(callback) {
var newFoo = { "name": "newFoo" };
Test.findOneAndUpdate(
{ "list.name": "foo" },
{
"$set": { "list.0": newFoo }
},
function(err,doc) {
console.log(doc);
callback();
}
)
},
function(callback) {
var newFoo = { "name": "moreFoo" };
Test.findOneAndUpdate(
{ "list.name": "newFoo" },
{
"$set": { "list.$": newFoo }
},
function(err,doc) {
console.log(doc);
callback();
}
);
}
]);
基本上就是你在做什么。现在仔细查看输出:
_id: 546419d60158d92a6f0167d2,
list:
[ { name: 'foo', _id: 546419d60158d92a6f0167d4 },
{ name: 'bar', _id: 546419d60158d92a6f0167d3 } ] }
{ _id: 546419d60158d92a6f0167d2,
__v: 0,
list:
[ { name: 'newFoo', _id: 546419d60158d92a6f0167d5 },
{ name: 'bar', _id: 546419d60158d92a6f0167d3 } ] }
{ _id: 546419d60158d92a6f0167d2,
__v: 0,
list:
[ { name: 'moreFoo' },
{ name: 'bar', _id: 546419d60158d92a6f0167d3 } ] }
这是三次更新的结果,正如你所说,在第三条语句之后,_id
确实丢失了。但请查看第二条语句中的 _id
值。它与原始“不同”,因此新的“子文档”“替换”了该元素,而不是更新“名称”。
这里肯定有一点“有趣”,但它更与“mongoose”如何翻译你的语句有关,而不是与 MongoDB 相关。 _id
值是由 mongoose 创建的,而不是 MongoDB 创建的。 MongoDB 仅为“顶级”文档创建一个 _id
值。 Mongoose 对数组执行此操作,因为它默认认为这是一件好事。
因此,在某些时候, Mongoose “计算出”您的“更新”实际上将“替换”指定索引处的元素,并根据架构规则创建一个新对象作为参数。
但如前所述,这不是$set
本来是要被使用的。如果您只想设置子文档的“名称”属性,而不设置其他属性,则可以使用 "dot notation" 指定该属性的“完整路径”。 .
这是修改后的列表,仅修改“名称”属性,没有其他任何内容:
var async = require('async'),
mongoose = require('mongoose'),
Schema = mongoose.Schema;
var testSchema = new Schema({
"list" : [{ "name": String }]
});
var Test = mongoose.model( 'Test', testSchema, "test" );
mongoose.connect('mongodb://localhost/test');
async.series([
function(callback) {
Test.remove({},callback);
},
function(callback) {
var test = new Test({ "list": [{ "name": "foo"},{ "name": "bar" }] });
test.save(function(err,doc) {
console.log(doc);
callback();
});
},
function(callback) {
var newFoo = { "name": "newFoo" };
Test.findOneAndUpdate(
{ "list.name": "foo" },
{
"$set": { "list.0.name": newFoo.name }
},
function(err,doc) {
console.log(doc);
callback();
}
)
},
function(callback) {
var newFoo = { "name": "moreFoo" };
Test.findOneAndUpdate(
{ "list.name": "newFoo" },
{
"$set": { "list.$.name": newFoo.name }
},
function(err,doc) {
console.log(doc);
callback();
}
);
}
]);
将其中的两个符号分别视为 "list.0.name"
和 "list.$.name"
。现在看看其中的区别:
{ __v: 0,
_id: 54641bcfdd89cf2c7299025e,
list:
[ { name: 'foo', _id: 54641bcfdd89cf2c72990260 },
{ name: 'bar', _id: 54641bcfdd89cf2c7299025f } ] }
{ _id: 54641bcfdd89cf2c7299025e,
__v: 0,
list:
[ { name: 'newFoo', _id: 54641bcfdd89cf2c72990260 },
{ name: 'bar', _id: 54641bcfdd89cf2c7299025f } ] }
{ _id: 54641bcfdd89cf2c7299025e,
__v: 0,
list:
[ { name: 'moreFoo', _id: 54641bcfdd89cf2c72990260 },
{ name: 'bar', _id: 54641bcfdd89cf2c7299025f } ] }
现在您会看到,在每种情况下,_id
都保持不变,这是因为“我们告诉更新不要管它”。
这基本上就是你想要的。使用 positional $
时匹配找到的索引位置 mongoose 的运算符“不应该”假设右侧缺少任何参数而没有 "dot notation"意味着应该创建一个新对象。相反,您“应该”指定您希望更改的“仅字段”的整个路径。
关于javascript - mongodb/mongoose - 使用 $ 运算符更新会丢失 _id,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26898022/
当我保存4条记录时,我需要将它们一一保存,所以我的代码是 a.save(function(){ b.save(function(){ c.save(function(){ d.save(f
Mongoose在版本4.2.7中创建了一个新的单个子文档功能(documentation和feature request),允许使用单个嵌入式子文档架构,其行为方式与一对多子文档的行为相同。 在父级
Mongoose 版本 >= 4.0 有一个时间戳选项,当 timestamps 设置为 时,该选项会为架构创建 updatedAt 和 createdAt 字段正确。 http://mongoose
我注意到某些图书馆喜欢 mock 鹅 ( https://github.com/mccormicka/Mockgoose/blob/master/test/index.spec.js ) 使用 req
我正在与 Mongoose 合作。我见过很多开发者发出以下命令: mongoose.Promise = global.Promise; 然后我很好奇 mongoose.Promise 的原始值是什么。
当我运行与数据库大量连接和断开连接的测试时,我收到以下警告。 (node) warning: possible EventEmitter memory leak detected. 11 connec
我可以在 Mongoose 的子文档数组中填充动态引用(使用“refPath”)虚拟字段吗? 数据结构如下 Group - Members -> User 代码:模型/模式 let MemberSc
我正在我的应用程序中做一些测试,看看内存缓存是否真的在工作。但是,由于 memory-cache 显然没有公开“命中”事件,我无法判断是否真的从缓存中获取数据。所以我试着看看当应用程序实际从数据库中获
我是 nestjs 的新手。我使用 @nestjs/mongoose,我需要在我的类模式中引用嵌套对象中的几个字段,但我不知道该怎么做。 dietDays 对象必须包含一个日期字段和包含对 Meal
我是 mongodb 的新手,我有一个这样的数据模型 { last_updated: Date.now(), csgo_items:[ {name: 'name', p
这是我的方案: var documentSchema = mongoose.Schema({ 'facts': [{ 'type': { type: String, requi
我想删除多个 _ids = ['123', '234', '345']; _ids.forEach(_id => { await model.deleteOne({ _id }); }); 有没有
我有一个像这样的 Mongoose 模式: var Address = { doorNo:String, city:String, state:String, coun
我的文档包含一个名为 clients 的字段那应该包含一组客户端ID。 { "first_name":"Nick", "last_name":"Parsons", "email":"nic
我遇到了以下我无法理解的代码行,尽管有很多教程提供了与 populate 的示例相关的信息。但没有一个能解释它究竟意味着什么。这是一个例子 var mongoose = require('mongoo
我有一个具有多个唯一性的架构,如下所示: var userSchema = new mongoose.Schema({ user: { type: String, unique:
我有一个 Mongoose 模式,其中有 4 个子模式。我一直在关注这里的文档https://github.com/LearnBoost/mongoose关于嵌入文档 var scenarios =
我希望每个字符串属性都默认设置为 true。有办法吗? ?? mongoose.Schema.String -> default { trim: true } var schema = new Sch
我有这个代码 var ClientSchema = new Schema({ name: {type: String, required: true, trim: true} }); var Cl
许多教程告诉您在您的 userSchema 页面中使用 bycrypt。保存新用户后,它会附带加密密码。伟大的。然而,我想,当我用某些东西编辑用户时,它也会重新哈希密码,导致无法登录。你能给我一个解决
我是一名优秀的程序员,十分优秀!