- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我们存储每个时间间隔的温度测量数据,我们希望只保留 90 个以前的数据。我们的数据结构是这样的:
{ "_id" : ObjectId("xxx"), "device" : "deviceId1", "count": 2, "values" :
[
{ "ts" : NumberLong("1471077454902"), "measureData" : 37.3 },
{ "ts" : NumberLong("1471077454911"), "measureData" : 37.4 }
]
}
count 是值的大小,例如当数组有 2 个元素时,则大小为 2。我们将 Java API 设计如下:
When the new measurement data comes:
Get corresponding device id count:
if count < 90:
Push the new mesaure data into values and increase count by 1
if count >90:
Pull the first element of the array and push the latest data into array.
Store the first element pulled into history collection.
是否有一个查询或一个聚合可以执行这些步骤?或者我们应该通过传统的方法来做到这一点,比如查询、评估,然后推送或拉/推。
////***************part 2 传统方法***************//
public class TSDesignMain {
public static void main(String[] args) {
MongoClient mClient = new MongoClient();
MongoDatabase db = mClient.getDatabase(MongoTSConstants.dbname);
long tWarmingStart = System.nanoTime();
InsertingWarmingDocument(db);
long tWarmingEnd = System.nanoTime();
double tDurationWarming = (double) (tWarmingEnd-tWarmingStart) / 1000/1000;
System.out.println("warming db, insert 10000 document per event duration is "+tDurationWarming+"ms");
long tInsert1Dev1DocStart = System.nanoTime();
for(int j=1;j<10000;j++){
storeMeasureDataIntoDB(db);
}
long tInsert1Dev1DocEnd = System.nanoTime();
double tInsert1Dev1DocDuration = (double) (tInsert1Dev1DocEnd-tInsert1Dev1DocStart) / 1000/1000;
System.out.println("insert 10000 document in 90*24*30 elements array duration is "+tInsert1Dev1DocDuration+"ms");
long tQuery1Dev1DocStart = System.nanoTime();
for(int j=1;j<10000;j++){
handleQueryDocument(db);
}
long tQuery1Dev1DocEnd = System.nanoTime();
double tQuery1Dev1DocDuration = (double) (tQuery1Dev1DocEnd-tQuery1Dev1DocStart) / 1000/1000;
System.out.println("query 10000 times in 90*24*30 elements array duration is "+tQuery1Dev1DocDuration+"ms");
mClient.close();
}
private static void InsertingWarmingDocument(MongoDatabase db) {
long ts = Calendar.getInstance().getTimeInMillis();
for(long i=1;i<100;i++){
db.getCollection(MongoTSConstants.tsDataPdCollection).insertOne(
new Document(MongoTSConstants.deviceFn,MongoTSConstants.deviceIdPrefix+i+"test")
.append(MongoTSConstants.tsFn,ts+i)
);
}
}
private static void handleQueryDocument(MongoDatabase db) {
Document where = new Document(MongoTSConstants.deviceFn,MongoTSConstants.deviceIdPrefix+1);
FindIterable<Document> it = db.getCollection(MongoTSConstants.tsDataInOneCollection).find(where);
MongoCursor<Document> cur = it.iterator();
int i=0;
if(!cur.hasNext()){
Document doc = cur.next();
ArrayList<Document> obj = (ArrayList<Document>) doc.get("values");
}
}
private static void storeMeasureDataIntoDB(MongoDatabase db) {
Document where = new Document(MongoTSConstants.deviceFn,MongoTSConstants.deviceIdPrefix+1);
FindIterable<Document> it = db.getCollection(MongoTSConstants.tsDataInOneCollection).find(where);
MongoCursor<Document> cur = it.iterator();
int i=0;
/**
* There is no device document in DB, insert new one. and use update to store 1st measure data
*
*/
if(!cur.hasNext()){
long tsInsert = System.nanoTime();
/**
* insert device id
*/
db.getCollection(MongoTSConstants.tsDataInOneCollection).insertOne(
new Document(MongoTSConstants.deviceFn,MongoTSConstants.deviceIdPrefix+1)
.append(MongoTSConstants.countFn,0));
/**
* using update to insert values array (store the first measure data)
*
*/
db.getCollection(MongoTSConstants.tsDataInOneCollection).updateOne(
new Document(MongoTSConstants.deviceFn,MongoTSConstants.deviceIdPrefix+1),
new Document("$push", new Document("values",new Document(MongoTSConstants.tsFn,tsInsert).
append(MongoTSConstants.measureDataFn,37.1)))
.append("$inc", new Document(MongoTSConstants.countFn,1)));
}else{
while(cur.hasNext()){
/**
* if i > 1, it means there are two doc with same device id, error!!
*/
if(i>=1){
//log error find two docs with same devicedID.
break;
}
Document doc = cur.next();
Integer count = doc.getInteger("count");
/**
* measure data has over 3 month.
*/
if( count >= MongoTSConstants.queryDataLength ){
/**
* get the first one in the array
*/
ArrayList<Document> obj = (ArrayList<Document>) doc.get("values");
Document doc1Elem = (Document) obj.get(0);
/**
* pull the first one in the array
*/
db.getCollection(MongoTSConstants.tsDataInOneCollection).updateOne(where,
new Document("$pop", -1));
long ts = Calendar.getInstance().getTimeInMillis();
/**
* push the new one
*/
db.getCollection(MongoTSConstants.tsDataInOneCollection).updateOne(where,
new Document("$push", new Document("values",new Document(MongoTSConstants.tsFn,ts).
append(MongoTSConstants.measureDataFn,37.8))));
//Store doc1Elem in another history collection;
db.getCollection(MongoTSConstants.tsHistoryCollection).insertOne(doc1Elem);
}else{
/**
* Measure data has not reach 3 month data
*/
long ts = Calendar.getInstance().getTimeInMillis();
db.getCollection(MongoTSConstants.tsDataInOneCollection).updateOne(where,
new Document("$push", new Document("values",new Document(MongoTSConstants.tsFn,ts).
append(MongoTSConstants.measureDataFn,37.9))).append("$inc", new Document(MongoTSConstants.countFn,1)));
}
i++;
}
}
}
}
最佳答案
在我看来,目前很难做到这一点(目前最新版本的mongodb是v3.2。),我不知道如何实现目标。
此处有 5 个操作,您想将它们合并为 1 个更新操作:
让我们把它们分成两部分:查询部分 & 更新部分
因为我不知道如何将Evaluation 放入更新语句的update action 部分:
db.test.update(
{/* Query criteria */},
{/* Update action */}
)
我采用了另一种方法 - 将评估放入查询条件部分,如下所示:
// We call this statement as UPDATE_STAT_1
db.test.update(
{device: 'deviceId1', count: {$lt: 90}},
{/* Update action: push new data to values and increase count */}
);
如果计数<90,您的示例文档将被匹配并执行更新操作(此时不关心更新操作是成功还是失败)。然后mongoDB会返回执行结果给你:
// Matched and updated
WriteResult({
"nMatched" : 1,
"nUpserted" : 0,
"nModified" : 1
});
结果告诉你一个文档被匹配和更新(nMatched = 1 & nModified = 1)
另一方面,如果count >= 90,则不会匹配和更新任何内容,执行结果将是:
// Not Matched
WriteResult({
"nMatched" : 0,
"nUpserted" : 0,
"nModified" : 0
})
所以你会知道计数达到90,你应该执行另一个更新语句:
// We call this statement as UPDATE_STAT_2
db.test.update(
{device: 'deviceId1', count: {$gte: 90}},
{/* Update action: push new data to values and pop the oldest*/}
);
总结,您可以先执行UPDATE_STAT_1,然后检查执行结果来决定是否需要像这样执行UPDATE_STAT_2(伪代码):
exe_result = run(UPDATE_STAT_1); // Run UPDATE_STAT_1
if(exe_result.nMatched == 0 && exe_result.nModified == 0) {
run(UPDATE_STAT_2); // Check the count and Run UPDATE_STAT_2
}
这是我合并查询(第 1 次)和评估(第 2 次)的方法。我不知道如何将上述所有代码合并到一条语句中。
我在本地试过,你可以把Push & Increase放在一条语句中:
db.test.update(
{device: 'deviceId1', count: {$lt: 90}},
{
$push: {values: { "ts" : "1471077454988", "measureData" : 39 }},
$inc : {count: 1}
}
);
同时进行 Push(3rd) 和 Increase(5th) 是合法的。但是如果你想压入和弹出数组:
db.test.update(
{device: 'deviceId1', count: {$gte: 90}},
{
$pop: {values: -1},
$push: {values: { "ts" : "1471077454911", "measureData" : 40 }}
}
);
你会得到错误:
WriteResult({
"nMatched" : 0,
"nUpserted" : 0,
"nModified" : 0,
"writeError" : {
"code" : 16837,
"errmsg" : "Cannot update 'values' and 'values' at the same time"
}
})
这意味着您不能同时对值进行推送和弹出操作。为此,我找到了这个解释:
The issue is that MongoDB doesn’t allow multiple operations on the same property in the same update call. This means that the two operations must happen in two individually atomic operations.
所以。总之,它无法将 Push(3rd)/Pop(4th)/Increase(5th) 合并到一个语句中。
根据上面的示例,Query(1st) 和 Evaluation(2nd) 不能合并,Push(3rd)/Pop(4th)/Increase(5th) 也不能合并。所以你可以用传统的方式来做这件事。谢谢。
关于mongodb - 蒙戈 : Storing time series measurement data,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38949796/
在尝试 time 的 python 执行时,我发现在一条语句中两次调用 time.time() 时出现奇怪的行为。在语句执行期间获取time.time() 有一个非常小的处理延迟。 例如time.ti
我要疯了。对于我的生活,我无法弄清楚为什么以下代码会导致 Unity 在我按下播放键后立即卡住。这是一个空的项目,脚本附加到一个空的游戏对象。在控制台中,什么也没有出现,甚至没有出现初始的 Debug
我要疯了。对于我的生活,我无法弄清楚为什么以下代码会导致 Unity 在我按下播放键后立即卡住。这是一个空的项目,脚本附加到一个空的游戏对象。在控制台中,什么也没有出现,甚至没有出现初始的 Debug
我不明白为什么下面的结果是一样的。我预计第一个结果是指针地址。 func print(t *time.Time) { fmt.Println(t) // 2009-11-10 23:00:00
Python 3.6.4 (v3.6.4:d48eceb, Dec 19 2017, 06:54:40) [MSC v.1900 64 bit (AMD64)] on win32 Type "help
当我有一个time.Time时: // January, 29th t, _ := time.Parse("2006-01-02", "2016-01-29") 如何获得代表 1 月 31 日的 ti
首先,我意识到不推荐使用 time with time zone。我要使用它是因为我将多个 time with time zone 值与我当前的系统时间进行比较,而不管是哪一天。 IE。用户说每天 0
长期以来,在 Rust 中精确测量时间的标准方法是 time crate 及其 time::precise_time_ns功能。但是,time crate 现在已被弃用,std 库有 std::tim
我正在我学校的一个科学集群上运行我的有限差分程序。该程序使用 openmpi 来并行化代码。 当程序连续运行时,我得到: real 78m40.592s user 78m34.920s s
尽管它们已被弃用并且有比 time 更好的模块(即 timeit),但我想知道这两个函数 time 之间的区别.clock() 和 time.time()。 从后者 (time.time()) 开始,
这个问题在这里已经有了答案: Python's time.clock() vs. time.time() accuracy? (16 个答案) 关闭 6 年前。 我认为两者都衡量时间量?但是他们返回
我正在尝试测试 http 请求处理代码块在我的 Flask Controller 中需要多长时间,这是我使用的示例代码: cancelled = [] t0 = time.time() t1 = ti
运行 python 的计算机时钟(Windows 或 Linux)时会发生什么自动更改并调用 time.time()? 我读到,当时钟手动更改为过去的某个值时,time.time() 的值会变小。 最
我有一个结构可能无法在其字段之一上设置 time.Time 值。测试无效性时,我不能使用 nil 或 0。time.Unix(0,0) 也不相同。我想到了这个: var emptyTime time.
我有一个打算用数据库记录填充的结构,其中一个日期时间列可以为空: type Reminder struct { Id int CreatedAt time.Time
问题陈述:通过匹配其百分比随机执行各种命令。比如执行 CommandA 50% 的时间和 commandB 25% 的时间和 commandC 15% 的时间等等,总百分比应该是 100%。 我的问题
我正在使用 laravel 6。我在同一个应用程序中有类似的 Controller 和类似的 View ,它工作正常。对比之后还是找不到错误。 Facade\Ignition\Exceptions\V
我需要用 ("%m/%d/%Y %H:%M:%S") 格式表示时间,我得到的浮点值是 time.time(). 我已经有了一个 time.time() 形式的值。例如,我已经有一个值,我每 0.3 秒
我正在使用以下方法获取 utc 日期时间: import datetime import time from pytz import timezone now_utc = datetime.datet
我在 Ubuntu 上使用 time.clock 和 time.time 为一段 python 代码计时: clock elapsed time: 8.770 s time elapsed time
我是一名优秀的程序员,十分优秀!