sql >> Databáze >  >> NoSQL >> MongoDB

Spring Data MongoDB agregace - shoda podle vypočtené hodnoty

Co potřebujete, je $redact operátor v agregačním rámci, který vám umožňuje zpracovat výše uvedenou logickou podmínku pomocí $cond operátor a používá speciální operace $$ PONECHAT "uchovat" dokument, kde je logická podmínka pravdivá nebo $$PRUNE k "odstranění" dokumentu, kde byla podmínka nepravdivá.

Tato operace je podobná jako u $projekt kanál, který vybere pole v kolekci a vytvoří nové pole, které obsahuje výsledek z dotazu na logickou podmínku a poté následný $match , kromě toho $redact používá jeden stupeň potrubí, který je efektivnější.

Zvažte následující příklad, který demonstruje výše uvedený koncept:

db.collection.aggregate([
    { 
        "$redact": {
            "$cond": [
                { 
                    "$gte": [
                        { "$divide": ["$Number1", "$Number2"] },
                        CONSTANT_VAR
                    ]
                },
                "$$KEEP",
                "$$PRUNE"
            ]
        }
    }
])

Protože není k dispozici žádná podpora pro $redact operátor zatím (v době psaní tohoto článku) bude řešením implementace AggregationOperation rozhraní, které zabalí operaci agregace do třídy, která má přijmout DBObject :

public class RedactAggregationOperation implements AggregationOperation {
    private DBObject operation;

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

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

které pak můžete použít v TypeAggregation :

Aggregation agg = newAggregation(
    new RedactAggregationOperation(
        new BasicDBObject(
            "$redact",
            new BasicDBObject(
                "$cond", new BasicDBObject()
                    .append("if", new BasicDBObject(
                        "$gte", Arrays.asList(
                            new BasicDBObject(
                                "$divide", Arrays.asList( "$Number1", "$Number2" )
                            ), 
                            CONSTANT_VAR
                        )
                    )
                )
                .append("then", "$$KEEP")
                .append("else", "$$PRUNE")
            )
        )
    )
);

AggregationResults<Example> results = mongoTemplate.aggregate(
    (TypedAggregation<Example>) agg, Example.class);

--AKTUALIZACE--

Následná reakce z komentářů ke kontrole nulových nebo nulových hodnot u Number2 pole v divizi, museli byste vnořit $cond výraz s logikou místo toho.

Následující příklad předpokládá, že máte hodnotu zástupného symbolu 1, pokud je buď Číslo2 neexistuje/je null nebo má hodnotu nula:

db.collection.aggregate([
    { 
        "$redact": {
            "$cond": [
                { 
                    "$gte": [
                        { 
                            "$divide": [
                                "$Number1", {
                                    "$cond": [
                                        {
                                            "$or": [
                                                { "$eq": ["$Number2", 0] },
                                                { 
                                                    "$eq": [
                                                        { "$ifNull": ["$Number2", 0] }, 0
                                                    ]
                                                }
                                            ]
                                        },
                                        1,  // placeholder value if Number2 is 0
                                        "$Number2"                                          
                                    ]
                                }
                            ] 
                        },
                        CONSTANT_VAR
                    ]
                },
                "$$KEEP",
                "$$PRUNE"
            ]
        }
    }
])

Ekvivalentní jarní agregace dat (netestováno)

Aggregation agg = newAggregation(
    new RedactAggregationOperation(
        new BasicDBObject(
            "$redact",
            new BasicDBObject(
                "$cond", Arrays.asList(
                    new BasicDBObject(
                        "$gte", Arrays.asList(
                            new BasicDBObject(
                                "$divide", Arrays.asList( 
                                    "$Number1", 
                                    new BasicDBObject(
                                        "$cond", Arrays.asList( 
                                            new BasicDBObject( "$or": Arrays.asList( 
                                                    new BasicDBObject("$eq", Arrays.asList("$Number2", 0)),
                                                    new BasicDBObject("$eq", Arrays.asList(
                                                            new BasicDBObject("$ifNull", Arrays.asList("$Number2", 0)), 0
                                                        )
                                                    )
                                                )
                                            ),
                                            1,  // placeholder value if Number2 is 0
                                            "$Number2"                                          
                                        )
                                    ) 
                                )
                            ), 
                            CONSTANT_VAR
                        )
                    ), "$$KEEP", "$$PRUNE"
                )
            )
        )
    )
);



  1. Jak implementuji tuto operaci dotazu a aktualizace mongodb (ovladač CSharp)?

  2. Jak nainstalovat php-mongo na CentOS 5.3?

  3. pole s jedinečnými hodnotami ve všech dokumentech jedné kolekce

  4. Operace aktualizace Mongodb uvnitř dílčího pole