- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我的第一个问题!我真的很想问对它,所以如果我可以问得更好,请帮助我改进它。
这是我发现的唯一一个似乎有点相关的问题,但我无法弄清楚如何将它与我正在尝试做的事情联系起来(他们的问题是特定于 JQuery 的;我的是特定于 Node.JS [尽管我找到了 EventEmitter 的浏览器版本并且也能够在浏览器中进行测试]): Run function once per event burst with jQuery
我有一个进程,我知道它会在一段时间内发出一连串的事件。
为了模拟这个过程,我编写了这段代码:
/*******************************************************/
/* This part taken directly from */
/* https://nodejs.org/api/events.html */
/* (with addition of "burstID") */
/* */ /* */
/* */ const EventEmitter = require('events'); /* */
/* */ /* */
/* */ class MyEmitter extends EventEmitter {} /* */
/* */ /* */
/* */ const myEmitter = new MyEmitter(); /* */
/* */ myEmitter.on('event', (burstID) => { /* */
/* */ console.log('an event occurred!', burstID); /* */
/* */ }); /* */
/* */ /* */
/*******************************************************/
const millisecondsToSustainBurst = 3000 ;
const millisecondsBetweenPossibleEventEmissions = 200 ;
const millisecondsUntilStartNextBurst = 5000 ;
const millisecondsUntilNoMoreBursts = 23000 ;
const now = new Date() ;
console.log('Time now: ' + now + '; should run until about ' + new Date(now.getTime() + millisecondsUntilNoMoreBursts)) ;
const doRandomEmitBurst = (startTimestamp, millisecondsToSustainBurst, burstID) => {
if (Math.random() > 0.5) myEmitter.emit('event', burstID) ;
if (
!((new Date()) - startTimestamp > millisecondsToSustainBurst)
) setTimeout(() => doRandomEmitBurst(startTimestamp, millisecondsToSustainBurst, burstID), millisecondsBetweenPossibleEventEmissions) ;
}
const doRegularRandomBursts = (startTimestamp, millisecondsUntilStartNextBurst, millisecondsUntilNoMoreBursts, callback) => {
if (
!((new Date()) - startTimestamp > millisecondsUntilNoMoreBursts)
) {
const now = new Date() ;
console.log('Time now: ' + now + '; starting random-event burst which will run for ' + (millisecondsToSustainBurst/1000) + ' seconds. ') ;
setTimeout(() => doRegularRandomBursts(startTimestamp, millisecondsUntilStartNextBurst, millisecondsUntilNoMoreBursts, callback), millisecondsUntilStartNextBurst) ;
doRandomEmitBurst(new Date(), millisecondsToSustainBurst, 'burstThatStartedAt' + now.getHours() + 'h' + now.getMinutes() + 'm' + now.getSeconds() + 's') ;
} else callback() ;
}
doRegularRandomBursts(new Date(), millisecondsUntilStartNextBurst, millisecondsUntilNoMoreBursts, () => console.log('Done at ' + (new Date()))) ;
const myBurstDetectedEmitter = new MyEmitter() ;
// NOW, what do I do HERE to say:
// I've seen a number of events occur in a 5-second period
// Now they've stopped
// Therefore I'm going to emit a different kind of event
现在,假设我想听听这些爆发的发生。
在我采取进一步行动之前,我想确保爆发已经结束。
我该怎么做?
首先,我可以像这样创建一个全局“var”(糟糕——我想避免使用可变变量):
var timeLastUpdated = {} ;
……然后……
function keepCheckingTimeLastUpdated(keyForUpdateCheck, callback) {
const timestampToCheckInOneSecond = (typeof timeLastUpdated[keyForUpdateCheck] !== 'undefined' ? timeLastUpdated[keyForUpdateCheck] : (new Date())) ;
setTimeout(() => {
console.log(
'checking if modifications to "' +
keyForUpdateCheck +
'" have occurred since ' +
timestampToCheckInOneSecond
) ;
if (timeLastUpdated[keyForUpdateCheck] === timestampToCheckInOneSecond) {
delete timeLastUpdated[keyForUpdateCheck] ;
callback() ;
}
else keepCheckingTimeLastUpdated(keyForUpdateCheck, callback) ;
}, 5000) ;
}
const makeNotificationHandler = () => (keyForUpdateCheck) => {
const timeNow = new Date() ;
if (typeof timeLastUpdated[keyForUpdateCheck] === 'undefined') keepCheckingTimeLastUpdated(keyForUpdateCheck, () => console.log(keyForUpdateCheck + ' changed')) ;
timeLastUpdated[keyForUpdateCheck] = timeNow ;
} ;
myEmitter.on('event', makeNotificationHandler()) ;
只是这似乎是一种反模式(我希望我使用的是正确的术语)。我的直觉告诉我,拥有一个全局对象在这里是错误的方法,并且有一个更像函数式编程的解决方案。
仅限感兴趣的人:
(出于回答问题的目的,请随意忽略)
增加了复杂性:在我的示例代码中,“burstID”永远不会相同,但在真实世界的示例中,它可能是相同的。我想等到自上次“burstID”出现以来经过了一定的时间,以便弄清楚变化的爆发是否真的完成了。
对于上下文,在实际应用程序中,我正在使用 node-postgres 在 PostGres 数据库上设置“LISTEN” . “burstID”在一个表中是主键,在其他多个表中也被用作外键。我正在监听所有使用共享 key 的表,并且我收到的消息包含此 key 。
最佳答案
在上述评论的帮助下回答我自己的问题。
一声大大的“谢谢!”至 Scott Sauyet感谢他的帮助和鼓励。
我决定创建一个我称之为“累加器”的东西:一个自动累加器。消息被路由到这些 Accumulomators,我认为他们是仓库里拿着箱子和秒表的人。
一旦 Accumulomator 被实例化,它就会开始查看秒表。每次秒表走到尽头时,累加器都会查看是否有与上次相同的消息堆。如果和上次一样,Accumulomator 会打包它的消息箱,将它们交给仓库,然后在阳光明媚的地方退休。
完整的更新代码如下。我欢迎任何修改以改进代码。
// To test in a browser, use:
// https://github.com/Olical/EventEmitter
// …in place of const EventEmitter = require('events');
/*******************************************************/
/* This part taken directly from */
/* https://nodejs.org/api/events.html */
/* (with addition of "burstID") */
/* */ /* */
/* */ const EventEmitter = require('events'); /* */
/* */ /* */
/* */ class MyEmitter extends EventEmitter {} /* */
/* */ /* */
/* */ const myEmitter = new MyEmitter(); /* */
/* */ myEmitter.on('event', (burstID) => { /* */
/* console.log('an event occurred!', burstID); */
/* */ }); /* */
/* */ /* */
/*******************************************************/
const millisecondsToSustainBurst = 3000 ;
const millisecondsBetweenPossibleEventEmissions = 200 ;
const millisecondsUntilStartNextBurst = 5000 ;
const millisecondsUntilNoMoreBursts = 23000 ;
const now = new Date() ;
console.log('Bursts starting. Time now: ' + now + '; should run until about ' + new Date(now.getTime() + millisecondsUntilNoMoreBursts)) ;
const doRandomEmitBurst = (startTimestamp, millisecondsToSustainBurst, burstID) => {
if (Math.random() > 0.5) myEmitter.emit('event', burstID) ;
if (
!((new Date()) - startTimestamp > millisecondsToSustainBurst)
) setTimeout(() => doRandomEmitBurst(startTimestamp, millisecondsToSustainBurst, burstID), millisecondsBetweenPossibleEventEmissions) ;
}
const doRegularRandomBursts = (startTimestamp, millisecondsUntilStartNextBurst, millisecondsUntilNoMoreBursts, callback) => {
if (
!((new Date()) - startTimestamp > millisecondsUntilNoMoreBursts)
) {
const now = new Date() ;
console.log('Time now: ' + now + '; starting random-event burst which will run for ' + (millisecondsToSustainBurst/1000) + ' seconds. ') ;
setTimeout(() => doRegularRandomBursts(startTimestamp, millisecondsUntilStartNextBurst, millisecondsUntilNoMoreBursts, callback), millisecondsUntilStartNextBurst) ;
doRandomEmitBurst(new Date(), millisecondsToSustainBurst, 'burstThatStartedAt' + now.getHours() + 'h' + now.getMinutes() + 'm' + now.getSeconds() + 's') ;
} else callback() ;
}
doRegularRandomBursts(new Date(), millisecondsUntilStartNextBurst, millisecondsUntilNoMoreBursts, () => console.log('Done at ' + (new Date()))) ;
const makeAccumulomator = (config) => {
if (typeof config !== 'object') throw new Error('Must specify configuration object.') ;
if (typeof config.callback !== 'function') throw new Error('Must specify callback function for when the end of new messages is reached.') ;
if (typeof config.millisecondsBetweenChecks !== 'number') throw new Error('Must specify milliseconds between checks.') ;
if (Number.isInteger(config.millisecondsBetweenChecks) === false) throw new Error('Must specify milliseconds between checks as an integer.') ;
if (typeof config.onStop !== 'function' && typeof config.onStop !== 'undefined') throw new Error('If defined at all, onStop must be a function.') ;
const accumulomator = {} ;
var accumulatedMessages = [] ;
var stop = false ;
const keepCheckingTimeLastUpdated = (callback) => {
const timestampToCheckInOneSecond = (accumulatedMessages.length > 0 ? accumulatedMessages[accumulatedMessages.length - 1].timestamp : (new Date())) ;
setTimeout(() => {
if (stop) { if (typeof config.onStop === 'function') config.onStop() ; }
else if (accumulatedMessages.length < 1) keepCheckingTimeLastUpdated(callback) ;
else if (accumulatedMessages[accumulatedMessages.length - 1].timestamp === timestampToCheckInOneSecond) { stop = true ; callback() ; }
else keepCheckingTimeLastUpdated(callback) ;
}, config.millisecondsBetweenChecks) ;
} ;
keepCheckingTimeLastUpdated(config.callback) ;
accumulomator.receiveMessage = (message) => accumulatedMessages.push({ message: message, timestamp: (new Date())}) ;
accumulomator.stopOnNextCheck = () => {
if (stop === true) throw new Error('Accumulomator is already stopped.') ;
else stop = true ;
}
accumulomator.isActive = () => stop === false ;
accumulomator.getAccumulatedMessages = () => accumulatedMessages ;
return accumulomator ;
}
const makeAccumulomatorWarehouse = (config) => {
if (typeof config !== 'object') throw new Error('Must specify configuration object.') ;
if (typeof config.callback !== 'function') throw new Error('Must specify callback function.') ;
if (typeof config.millisecondsBetweenChecks !== 'number') throw new Error('Must specify milliseconds between checks.') ;
if (Number.isInteger(config.millisecondsBetweenChecks) === false) throw new Error('Must specify milliseconds between checks as an integer.') ;
if (typeof config.messageRouter !== 'function') throw new Error('Must specify message router function.') ;
if (typeof config.sendCallbackAccumulatedMessages !== 'undefined') if (typeof config.sendCallbackAccumulatedMessages !== 'boolean') throw new Error('Must specify whether or not to send callback accumulated messages as a boolean (if unspecified, accumulated messages will not be included).') ;
if (typeof config.onAccumulomatorStop !== 'function' && typeof config.onAccumulomatorStop !== 'undefined') throw new Error('If defined at all, onAccumulomatorStop must be a function.') ;
var sendCallbackAccumulatedMessages = false ;
if (typeof config.sendCallbackAccumulatedMessages !== 'undefined') sendCallbackAccumulatedMessages = config.sendCallbackAccumulatedMessages ;
const accumulomatorWarehouse = {} ;
const accumulomators = {} ;
var warehouseIsShuttingDown = false ;
accumulomatorWarehouse.receiveMessage = (message) => {
accumulomatorName = config.messageRouter(message) ;
if (typeof accumulomatorName !== 'string') throw new Error('The value returned from messageRouter must be a string with a unique identifier.') ;
if (typeof accumulomators[accumulomatorName] === 'object') if (accumulomators[accumulomatorName].isActive() === false) delete accumulomators[accumulomatorName] ;
if (typeof accumulomators[accumulomatorName] === 'undefined') {
if (warehouseIsShuttingDown === false) accumulomators[accumulomatorName] = makeAccumulomator({
callback: () => config.callback(
(() => {
objectToReturn = {} ;
objectToReturn.key = accumulomatorName ;
if (sendCallbackAccumulatedMessages === true) objectToReturn.messages = accumulomators[accumulomatorName].getAccumulatedMessages() ;
return objectToReturn ;
})()
) ,
millisecondsBetweenChecks: config.millisecondsBetweenChecks ,
onStop: () => config.onAccumulomatorStop(accumulomatorName)
}) ;
}
if (typeof accumulomators[accumulomatorName] === 'object') accumulomators[accumulomatorName].receiveMessage(message) ;
}
periodicallyRetireAccumulomators = () => {
Object.keys(accumulomators).forEach((accumulomator) => {
if (accumulomators[accumulomator].isActive() === false) delete accumulomators[accumulomator] ;
}) ;
if (!(warehouseIsShuttingDown === true && Object.keys(accumulomators).length === 0)) setTimeout(periodicallyRetireAccumulomators, 10000) ;
} ;
periodicallyRetireAccumulomators() ;
accumulomatorWarehouse.shutDownWarehouse = () => {
Object.keys(accumulomators).forEach((accumulomator) => {
if (accumulomators[accumulomator].isActive() === true) accumulomators[accumulomator].stopOnNextCheck() ;
}) ;
warehouseIsShuttingDown = true ;
}
return accumulomatorWarehouse ;
}
myAccumulomatorWarehouse = makeAccumulomatorWarehouse({
callback: (accumulomatorWarehousePackage) => console.log('Done with accumulomator.', accumulomatorWarehousePackage.key, accumulomatorWarehousePackage.messages) ,
millisecondsBetweenChecks: 2000 ,
messageRouter: (message) => message ,
sendCallbackAccumulatedMessages: true ,
onAccumulomatorStop: (accumulomatorName) => console.log('Accumulomator for ' + accumulomatorName + ' manually stopped')
}) ;
myEmitter.on('event', myAccumulomatorWarehouse.receiveMessage) ;
setTimeout(myAccumulomatorWarehouse.shutDownWarehouse, 10000) ;
关于javascript - 如何检测一连串相关事件已完成,然后发出一个事件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41916451/
第一段代码工作正常,并给出了我需要的结果。我现在想做的是让它在 'as num' 上返回 3 个数字值对于“as num”上的 3 个不同值,对于同一列上的 3 个不同位置 SELEC
我想分析一些数据以编写定价算法。以下日期可用: 我需要三个变量/维度的函数/相关因子,它显示三个维度(pers_capacity、卧室数量、浴室数量)增长时中位数(价格)的变化。例如Y(#pers_c
正如标题所说 - 我的 Sprite Kit 游戏时不时地在后台崩溃,总是出现此错误 - Exception Type: EXC_BAD_ACCESS (SIGSEGV) Exception Sub
假设我尝试保存以下数据,并且Songs模型的name属性上设置了Phalcon \ Mvc \ Model \ Validator \ PresenceOf验证器 // Get an existing
我有一个 if 控件,如下所示; if (Directory.Exists(System.IO.Path.Combine(systemPath, "Reports", companyName))
有人可以告诉我我们使用 ReadLine() 从文件 (.txt) 中读取特定行吗?现在我想读取文件的全部内容(不仅仅是第一行)。为此我需要使用什么方法。我用谷歌搜索了很多,但找不到解决方案。 我的代
我相信在大学时我用从 C 派生的语言为 FPGA 编写了一个程序。我了解 VHDL 和 verilog 等语言。但是,我不明白的是程序员在使用哪个方面有多少选择?它依赖于FPGA吗?我将使用 Xili
我有一个 if 控件,如下所示; if (Directory.Exists(System.IO.Path.Combine(systemPath, "Reports", companyName))
如何在运行时更改 Dashcode (Javascript) 中图像对象的源? 我试过: var image = document.getElementById("image").object;ima
我有几个相互关联的类,它们将被多种不同的算法使用 例子: struct B; struct A { B* parent; }; struct B { std::vector child
我正在开发一个网站,用户在客户收到的表中输入金额,如果任何客户没有提供分期付款(金额),则用户不会在表中输入任何金额,并且用户希望获取违约者的信息客户以10天为基础。所以我的问题是应该定义什么表和属性
我试图从上一个条目中选择一个值,并每次将该数字加一。我让它工作到选择当前条目值(默认 1000)并递增 1 并重新插入该值(因此每次最终都是 1001)。我需要它来选择该字段的最后一个条目,这样它将变
我不擅长“制作”查询。假设这是我的数据库: artist pics ------------------- -
最近,我要为我的网站做一个即时通知系统。我听说 COMET 在这种情况下必不可少。 我已经搜索 PHP 和 Comet 一段时间了,但是,我发现的指南和文章似乎只是循环中的 ajax 请求。例如,有一
我正在开发一款 iOS 游戏,我希望 clown 在场景外生成,然后向下移动。我的想法是全部创建它们,并将它们以 360 像素的距离放置在不可见的场景中。 像这样: SKSpriteNode *clo
我有以下子订单表。 mysql> select * from suborder; +-------------+------------------+ | order_state | bookin
这可能是一个有点初学者的问题,但考虑到在 Java 中调试编码是相当相关的:什么时候编码与 String 对象相关? 假设我有一个要保存到文件中的字符串对象。 String 对象本身是否使用某种我应该
首先我想说我是 CPP 的新手(我从 cpp11 开始):)考虑以下实体:学生(名字+姓氏)和组(描述+更多学生)。我在 C++ 中创建了以下 2 个类: class Student { privat
我正在尝试在单击该复选框时同步更新我的数据库。我决定使用 aJax,但它似乎无法识别 ajax。 代码:将成为 Switch_Active(this.id) 函数的元素 ... Deactivat
我正在创建一个菜单。菜单如下。 $('.category').mouseover(function() { $(this).removeClass('category').addClass('cate
我是一名优秀的程序员,十分优秀!