Прямая копия объекта документа mongodb - PullRequest
0 голосов
/ 14 декабря 2018

Короче говоря, я работаю внутри гигантской противной рекурсивной функции, которая создает объект Document mongo.Я должен расширить функциональность этой функции для поддержки сложных запросов и должен знать, как создать прямую копию (а не ссылку на объект) объекта Document mongo.Есть ли способ сделать это?Текущая реализация делает следующее:

private Document generateArrayAddsDoc(DataChanges dataChanges, Document filter)
    {
        Document arrayUpdatesDoc = new Document();
        generateArrayDocImpl(arrayUpdatesDoc, null, dataChanges, DataChanges.Type.MODEL_ADD, filter, 0);
        return arrayUpdatesDoc;
    }

Вот противная рекурсивная функция только для контекста:

private void generateArrayDocImpl(Document arrayUpdateDoc, String parentFieldName, DataChanges dataChanges, DataChanges.Type type, Document filter, int currentLevel)
    {
        switch (dataChanges.getType())
        {
            case MODEL_UPDATE:
            {
                for (DataChange dataChange : dataChanges.getDataChanges())
                {
                    if (dataChange.getNewValue() instanceof DataChanges)
                    {
                        DataChanges arrayChanges = (DataChanges) dataChange.getNewValue();

                        Document arrayDoc = new Document();
                        String dotFieldName = generateDotFieldName(parentFieldName, dataChange.getFieldName());

                        if (type == DataChanges.Type.MODEL_ADD || type == DataChanges.Type.MODEL_DELETE)
                        {
                            generateArrayDocImpl(arrayDoc, dotFieldName, arrayChanges, type, filter, currentLevel+1);
                        }

                        if (!arrayDoc.isEmpty())
                        {
                            if (currentLevel > 0)
                            {
                                arrayUpdateDoc.append(dotFieldName, arrayDoc);
                            }
                            else
                            {
                                arrayUpdateDoc = arrayDoc; // need to direct copy instead of referencing here
                            }
                        }
                    }
                }

                break;
            }
            case ARRAY_UPDATE:
            {
                List<Document> updateDocsList = new ArrayList<>();
                for (DataChange dataChange : dataChanges.getDataChanges())
                {
                    Document arrayElementDoc = null;
                    DataChanges arrayElementChanges = (DataChanges) dataChange.getNewValue();

                    if (arrayElementChanges.getType() == type)
                    {
                        arrayElementDoc = generateFieldUpdatesDoc(arrayElementChanges, filter);
                    }
                    else if (arrayElementChanges.getType() == DataChanges.Type.MODEL_UPDATE)
                    {
                        String dotFieldName = generateDollarSignIndexedFieldName(parentFieldName, dataChange.getFieldName());
                        generateArrayDocImpl(arrayUpdateDoc, dotFieldName, arrayElementChanges, type, filter, currentLevel+1);
                    }

                    if (null != arrayElementDoc && !arrayElementDoc.isEmpty())
                    {
                        updateDocsList.add(arrayElementDoc);
                    }
                }

                if (!updateDocsList.isEmpty() && type == DataChanges.Type.MODEL_ADD)
                {
                    arrayUpdateDoc.append("$each", updateDocsList);
                }

                if (!updateDocsList.isEmpty() && type == DataChanges.Type.MODEL_DELETE)
                {
                    String fieldName = pullFieldNameFromList(updateDocsList);
                    List<String> valuesToDelete = getValuesToDelete(updateDocsList);

                    Document doc = new Document();
                    doc.append("$in", valuesToDelete);

                    arrayUpdateDoc.append(fieldName, doc);
                }

                break;
            }
        }
    }

Где эта функция выполняет "arrayUpdateDoc = arrayDoc;", мне нужноскопируйте документ в новый объект, поскольку arrayDoc уничтожается после завершения вызова этой функции, что приводит к ошибке.Будем весьма благодарны за любые рекомендации относительно создания прямого объекта (не по ссылке) для объекта «Монго».

1 Ответ

0 голосов
/ 14 декабря 2018

Вы можете получить карту ввода документов и передать ее конструктору при создании нового документа:

Document original = someDocument;
Document clone = new Document(original.entrySet());

В вашем случае:

arrayUpdateDoc = new Document(arrayDoc.entrySet());

// Редактировать:

Я понял это неправильно.Однако Документ реализует Map<String, Object>.Это означает, что должно работать следующее:

arrayUpdateDoc = new Document(arrayDoc);

// Другое редактирование:

То, что вы пытаетесь сделать, не будет работать в Java.Например:

public static void main(String[] args)
{
    String test = "test";
    System.out.println(test);
    someMethod(test);
    System.out.println(test);
}

public static void someMethod(String test)
{
    test = "changed";
}

вывод:

test
test

Ссылка вызывающего абонента не будет обновлена ​​для нового объекта.Ссылка копируется при вызове метода.Итак, насколько я знаю, вам понадобится тип возвращаемого значения для вашего метода.

...