sql >> Databasteknik >  >> NoSQL >> MongoDB

slå upp i mongodb-aggregation

Eftersom du har kapslade arrayer måste du använda $unwind operatör först för att avnormalisera de inbäddade dokumenten innan du använder $lookup pipeline (såvida du inte redan har förenklat dem i din aggregeringsoperation):

db.personaddress.aggregate([
    { "$unwind": "$address" },
    { "$unwind": "$address.location" },
    {
        "$lookup": {
            "from": "places", 
            "localField": "address.location.place._id", 
            "foreignField": "_id", 
            "as": "address.location.place", 
        }
    }
])

som sedan kan implementeras som (otestad):

LookupOperation lookupOperation = LookupOperation.newLookup()
    .from("places")
    .localField("address.location.place._id")
    .foreignField("_id")
    .as("address.location.place");

Aggregation agg = newAggregation(
    unwind("address"),
    unwind("address.location"),
    lookupOperation  
);

AggregationResults<OutputDocument> aggResults = mongoTemplate.aggregate(
    agg, PersonAddressDocument.class, OutputDocument.class
);

Om din Spring Data-version inte stöder detta är en lösning att implementera AggregationOperation gränssnitt för att ta in ett DBObject :

public class CustomGroupOperation implements AggregationOperation {
    private DBObject operation;

    public CustomGroupOperation (DBObject operation) {
        this.operation = operation;
    }

    @Override
    public DBObject toDBObject(AggregationOperationContext context) {
        return context.getMappedObject(operation);
    }
}

Implementera sedan $lookup drift som ett DBO-objekt i aggregeringspipelinen:

DBObject lookupOperation = (DBObject)new BasicDBObject(
    "$lookup", new BasicDBObject("from", "places")
        .append("localField", "address.location.place._id")
        .append("foreignField", "_id")
        .append("as", "address.location.place")       
);

som du sedan kan använda som:

Aggregation agg = newAggregation(
    unwind("address"),
    unwind("address.location"),
    lookupOperation  
);

AggregationResults<OutputDocument> aggResults = mongoTemplate.aggregate(
    agg, PersonAddressDocument.class, OutputDocument.class
);


  1. Hur aggregerar man efter datum när en fullständig tidsstämpel ges i aggregeringsramverket?

  2. Node.js &Redis; Väntar på att en loop ska sluta

  3. Django ValueError:Ingen rutt hittades för sökväg 'ws/chat//'

  4. Kontrollera att fältet finns med MongoDB