我有以下.js文件,node.js APP的一部分,以及2个问题。
var User = require('../app/models/user');
var Agenda = require('agenda');
var mongoConnectionString = "mongodb://localhost/agenda";
var agenda = new Agenda({db: {address: mongoConnectionString}});
agenda.define('handle energy', function(job, done) {
let lastUpdateRun = new Date(Date.now() - 50*1000); // 50 seconds ago for some buffer room
/****** Energy Loss Per Minute ******/
User.update( // Over 85 -> ~ 40 loss / hour
{'local.energy' : { "$gt" : 85}, "local.energyUpdatedAt": {"$lt": lastUpdateRun}}, // query
{"$inc": { "local.energy": -0.66 }, "local.energyUpdatedAt": new Date() } , // update
{"multi": true}, // Options
function(err, result) { // Callback
if(err) {
console.log(err)
}
}
);
User.update( // Over Under 80 -> ~ 25 loss / hour
{'local.energy' : { "$gt": 50, "$lte" : 85 }, "local.energyUpdatedAt": {"$lt": lastUpdateRun}}, // query
{"$inc": { "local.energy": -0.42 }, "local.energyUpdatedAt": new Date() } , // update
{"multi": true}, // Options
function(err, result) { // Callback
if(err) {
console.log(err)
}
}
);
User.update( // Under 50 -> ~ 15 loss / hour
{'local.energy' : { "$gt": 25, "$lte" : 50}, "local.energyUpdatedAt": {"$lt": lastUpdateRun}}, // query
{"$inc": { "local.energy": -0.25 }, "local.energyUpdatedAt": new Date() }, // update
{"multi": true}, // Options
function(err, result) { // Callback
if(err) {
console.log(err)
}
}
);
User.update( // Under 25 -> ~ 10 loss / hour
{'local.energy' : { "$gt": 10, "$lte" : 25}, "$or" :[{'local.estate.owned.movedIn' : false}, {'local.estate.rented.movedIn' : false}], "local.energyUpdatedAt": {"$lt": lastUpdateRun} }, // query
{"$inc": { "local.energy": -0.167 }, "local.energyUpdatedAt": new Date()} , // update
{"multi": true}, // Options
function(err, result) { // Callback
if(err) {
console.log(err)
}
}
);
/******* Energy Gain Per Minute, No Office *******/
User.update( // Set Energy to Ten if over 9.5 and under 10
{'local.energy' : { "$lt": 10, "$gte" : 9.9}, "$or" : [{'local.estate.movedIn' : false}, {'local.estate.rented.movedIn' : false}], "local.energyUpdatedAt": {"$lt": lastUpdateRun} }, // query
{"$set": { "local.energy": 10 }, "local.energyUpdatedAt": new Date() } , // update
{"multi": true}, // Options
function(err, result) { // Callback
if(err) {
console.log(err)
}
}
);
User.update( // Under 10 -> ~ 5 gain / hour
{'local.energy' : { "$lt": 9.9}, "$or" : [{'local.estate.movedIn' : false}, {'local.estate.rented.movedIn' : false}], "local.energyUpdatedAt": {"$lt": lastUpdateRun} }, // query
{"$inc": { "local.energy": 0.085 }, "local.energyUpdatedAt": new Date() } , // update
{"multi": true}, // Options
function(err, result) { // Callback
if(err) {
console.log(err)
}
}
);
/******* Energy Gain Per Minute, With Office *******/
User.update( // Set Energy to 25 if over 24.5 and under 10
{'local.energy' : { "$lt": 10, "$gte" : 24.5}, "$or" : [{'local.estate.movedIn' : true}, {'local.estate.rented.movedIn' : true}], "local.energyUpdatedAt": {"$lt": lastUpdateRun} }, // query
{"$set": { "local.energy": 10 }, "local.energyUpdatedAt": new Date() } , // update
{"multi": true}, // Options
function(err, result) { // Callback
if(err) {
console.log(err)
}
}
);
User.update( // Under 25 -> ~ 15 gain / hour
{'local.energy' : { "$lt": 24.5 } , "$or" : [{'local.estate.movedIn' : true}, {'local.estate.rented.movedIn' : true}], "local.energyUpdatedAt": {"$lt": lastUpdateRun}}, // query
{"$inc": { "local.energy": 0.250 }, "local.energyUpdatedAt": new Date() } , // update
{"multi": true}, // Options
function(err, result) { // Callback
if(err) {
console.log(err)
}
}
);
done()
});
agenda.on('ready', function() {
agenda.every('one minute', 'handle energy');
agenda.start();
});
// Handle Office Rent Expiration Every Hour
//Handle A Few other things at different times
问题#1: 有没有更好的方法来处理所有这些更新,我在短时间内有相当多的更新,我一遍又一遍地重复90%的代码。此外,性能方面是这样的,或者我犯了一个大错误?
问题#2: 如果我需要每小时更多计划任务1每25分钟一次,我应该为每个任务创建一个setInterval,因为iirc javascript不处理多个setIntervals。那我该怎么处理呢?
P.S。我还在学习node / js所以如果这些都是愚蠢的问题,请不要讨厌我。谢谢你的耐心等待。
答案 0 :(得分:2)
更新很好。如果他们需要完成,他们需要完成。鉴于它们是重复运行的,只需确保在local.energy
和您要查询的每个其他字段之间添加一些复合索引:
UserSchema.index({"local.energy": 1});
UserSchema.index({"local.energy": 1, "local.estate.owned.movedIn": 1});
UserSchema.index({"local.energy": 1, "local.estate.rented.movedIn": 1});
//etc.
我唯一看到的关于逻辑的问题是你的更新是并行运行的,没有对重复更新的限制。想象一下,你有一份文件:
DocA = {
"local.energy": 9.85
}
取决于两个能源收益/无办公室内的执行顺序"查询,你可以有不同的结果。
{"local.energy": {$lt: 9.9}}
的更新查询可能会将DocA增加到9.935。现在它属于{"local.energy": {$gte: 9.9, $lt: 10}}
的更新查询,它会将其设置为10
。我建议在架构(和索引)中添加"lastUpdatedAt": Date
字段,并使用它限制查询:
let lastUpdateRun = new Date(Date.now() - 50*1000); // 50 seconds ago for some buffer room
User.update( // Over 85 -> ~ 40 loss / hour
{'local.energy' : { "$gt" : 85}, "lastUpdatedAt": {"$lt": lastUpdateRun}}, // query
{"$inc": { "local.energy": -0.66 }, "lastUpdatedAt": new Date()}, // update
{"multi": true}, // Options
function(err, result) { // Callback
if(err) {
console.log(err)
}
}
);
这样您只会更新在最近一段时间内未更新的内容。