- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个程序,我从服务器请求天气数据,处理数据,然后使用 mongoose 将其保存到 mlab 帐户。我正在收集 10 年的数据,但我请求数据的 API 一次只允许请求大约一年的数据。
我正在使用 findOndAndUpdate 创建/更新每个气象站的文档,但在更新数据对象中的数组时遇到问题。 (可能不是描述它的最佳方式......)
例如,这是模型:
const stnDataSchema = new Schema(
{
station: { type: String, default: null },
elevation: { type: String, default: null },
timeZone: { type: String, default: null },
dates: {},
data: {}
},
{ collection: 'stndata' },
{ runSettersOnQuery: true }
)
日期对象如下所示:
dates: ["2007-01-01",
"2007-01-02",
"2007-01-03",
"2007-01-04",
"2007-01-05",
"2007-01-06",
"2007-01-07",
"2007-01-08",
"2007-01-09"]
数据对象如下:
"data": [
{
"maxT": [
0,
null,
4.4,
0,
-2.7,
etc.....
我想要发生的是,当我运行 findOneAndUpdate 时,我想根据站点查找文档,然后将新的 maxT 值和日期附加到相应的数组中。我让它适用于日期数组,但由于我更新的元素是嵌套的,因此数据数组遇到了麻烦。我试过这个:
const update = {
$set: {'station': station, 'elevation': elevation, 'timeZone': timeZone},
$push: {'dates': datesTest, 'data.0.maxT': testMaxT}};
StnData.findOneAndUpdate( query, update, {upsert: true} ,
function(err, doc) {
if (err) {
console.log("error in updateStation", err)
throw new Error('error in updateStation')
}
else {
console.log('saved')
但在 mlab 中得到如下输出:
"data": {
"0": {
"maxT": [
"a",
"b",
问题是我得到的是“0”而不是一个元素的数组。我尝试了“data[0].maxT”,但是当我这样做时什么也没有发生。
问题是,第一次运行站的数据时,我想创建一个新文档,其中包含第三个代码块中格式的数据对象,然后在后续运行中,一旦该文档已经存在,就更新具有新值的 maxT 数组。有什么想法吗?
最佳答案
您将得到以下输出:
"data": {
"0": {
"maxT": [
"a",
"b",
因为您正在更新插入文档。处理文档数组时,更新插入会变得有点复杂。
更新数组时,MongoDB 知道 data.0
指的是数组中的第一个元素。然而,在插入时,MongoDB 无法判断它是数组还是对象。所以它假设它是一个对象。因此,它不是插入 ["val"]
,而是插入 {"0": "val"}
。
最简单的解决方案
不要使用更新插入。为每个新气象站插入一个文档,然后使用 findOndAndUpdate
将值推送到文档中的数组中。只要你第一次正确插入数组,你就能够推送它们,而不会让它们变成对象。
如果 data
仅包含一个对象,则另一种简单解决方案
从您的问题来看,您在 data
中似乎只有一个对象。如果是这种情况,您可以将 maxT
数组设为顶级,而不是作为数组中单个文档的属性。那么它的行为就像 dates
。
更复杂的 MongoDB 3.6 解决方案
如果您确实离不开更新插入,MongoDB 3.6 引入了过滤位置运算符 $[<identifier>]
。您可以使用此运算符来更新数组中与查询匹配的特定元素。 Unlike the simple positional operator $
,只要使用精确匹配,新的 $[<identifier>]
运算符就可以用于更新插入。
您可以在此处阅读有关此运算符的更多信息:https://docs.mongodb.com/manual/reference/operator/update/positional-filtered/
因此,您的 data
对象需要有一个可以完全匹配的字段(例如 name
)。示例查询如下所示:
let query = {
_id: 'idOfDocument',
data: [{name: 'subobjectName'}] // Need this for an exact match
}
let update = {$push: {'data.$[el].maxT': testMaxT}}
let options = {upsert: true, arrayFilters: [{'el.name': 'subobjectName'}]}
StnData.findOneAndUpdate(query, update, options, callbackFn)
正如您所看到的,这增加了更多的复杂性。忘记尝试进行更新插入会更容易。只需执行一次插入然后更新即可。
此外mLab目前不支持MongoDB 3.6。因此,在支持 3.6 之前,使用 mLab 时此方法不可行。
关于javascript - Mongoose findOneAndUpdate : create and then update nested array,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49203915/
当我保存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。保存新用户后,它会附带加密密码。伟大的。然而,我想,当我用某些东西编辑用户时,它也会重新哈希密码,导致无法登录。你能给我一个解决
我是一名优秀的程序员,十分优秀!