我正在使用mongo-java-driver 3.0.2.
我有一个MongoCollection.aggregate(List<Bson> pipeline)用于排序和限制的方法:
private static MongoIterable<Document> selectTop(int n) {
BasicDBObject sortFields = new BasicDBObject("score", -1);
BasicDBObject sort = new BasicDBObject("$sort", sortFields);
BasicDBObject limit = new BasicDBObject("$limit", n);
List<BasicDBObject> pipeline = new ArrayList<>();
pipeline.add(sort);
pipeline.add(limit);
return playersCollection.aggregate(pipeline);
}
Run Code Online (Sandbox Code Playgroud)
什么时候n很大,它失败了:
com.mongodb.MongoCommandException: Command failed with error 16820: 'exception: Sort exceeded memory limit of 104857600 bytes, but did not opt in to external sorting. Aborting operation. Pass allowDiskUse:true to opt in.'
Run Code Online (Sandbox Code Playgroud)
我发现MongoDB shell提供了一个方法db.collection.aggregate(pipeline, options)(链接),其中options可以包含一个allowDiskUse字段.
我在Java API中找不到与此相当的东西.虽然有一个AggregationOptions类,但MongoCollection该类不提供aggregate(List<Bson> pipeline, AggregationOptions options)方法.
Bla*_*ven 11
这仍然适用于3.0.3驱动程序:
MongoClient client = new MongoClient(new ServerAddress("127.0.0.1", 27017));
DB test = client.getDB("test");
DBCollection sample = test.getCollection("sample");
List<DBObject> aggregationQuery = Arrays.<DBObject>asList(
new BasicDBObject("$sort",new BasicDBObject("score",-1)),
new BasicDBObject("$limit",1)
);
System.out.println(aggregationQuery);
Cursor aggregateOutput = sample.aggregate(
aggregationQuery,
AggregationOptions.builder()
.allowDiskUse(true)
.build()
);
while ( aggregateOutput.hasNext() ) {
DBObject doc = aggregateOutput.next();
System.out.println(doc);
}
Run Code Online (Sandbox Code Playgroud)
当然你也可以使用更新的类:
MongoClient client = new MongoClient(new ServerAddress("192.168.2.4", 27017));
MongoDatabase db = client.getDatabase("test");
MongoCollection<Document> collection = db.getCollection("sample");
AggregateIterable<Document> result = collection.aggregate(Arrays.asList(
new BasicDBObject("$sort", new BasicDBObject("score", -1)),
new BasicDBObject("$limit", 1)
)).allowDiskUse(true);
MongoCursor<Document> cursor = result.iterator();
while (cursor.hasNext()) {
Document doc = cursor.next();
System.out.println(doc);
}
Run Code Online (Sandbox Code Playgroud)
所以.aggregate()在MongoCollection上返回一个AggregateIterable类实例,它有一个.allowDiskuse()方法和其他方法来设置聚合选项.