2

了解しました。SOユーザー。私はCouchDBを学び、使用しようとしています。StackExchangeデータのエクスポートをXMLファイルから行ごとにドキュメントとしてロードしているので、Couch内のドキュメントは基本的に次のようになります。

//This is a representation of a question:
{
 "Id" : "1",
 "PostTypeId" : "1",
 "Body" : "..."
}

//This is a representation of an answer
{
 "Id" : "1234",
 "ParentId" : "1",
 "PostTypeId" : "2"
 "Body" : "..."
}

(これらのドキュメントのインポートでは基本的にすべての属性がテキストとして扱われるという事実を無視してください。実数やブールなどを使用すると、スペース/処理効率が向上することを理解しています。)

私がやりたいのは、これを単一の集約ドキュメントにマップすることです。

これが私の地図です:

function(doc) {
    if(doc.PostTypeId === "2"){
      emit(doc.ParentId, doc);
    }
    else{
        emit(doc.Id, doc);
    }
}

そして、ここに削減があります:

function(keys, values, rereduce){
    var retval = {question: null, answers : []};

    if(rereduce){
        for(var i in values){
            var current = values[i];
            retval.answers = retval.answers.concat(current.answers);
            if(retval.question === null && current.question !== null){
                retval.question = current.question;
            }
        }
    }
    else{
        for(var i in values){
            var current = values[i];            

            if(current.PostTypeId === "2"){
                retval.push(current);
            }
            else{
                retval.question = current;
            }
        }
    }
    return retval;
}

理論的には、これにより次のようなドキュメントが生成されます。

{
    "question" : {...},
    "answers" : [answer1, answer2, answer3]
}

しかし、代わりに、標準の「十分に速く減少しない」エラーが発生します。

Map-Reduceを誤って使用していますか?CouchDbでこれを実現する方法について確立されたパターンはありますか?

(質問が「親」であり、回答がIDだけでなく「子供」である、完全なドキュメントでの応答を希望することにも注意してください。)

4

1 に答える 1

3

したがって、上記でやろうとしていることを達成するための「正しい」方法は、デザイン ドキュメントの一部として「リスト」を追加することです。(そして、私が達成しようとしている目的は、「文書の照合」と呼ばれているようです)。

いずれにせよ、マップを好きなように構成し、それを同じ関数内の「リスト」と組み合わせることができます。

上記の質問を解決するために、reduce (map 関数のみ) を削除し、次のような関数を追加しました。

{
   "_id": "_design/posts",
   "_rev": "11-8103b7f3bd2552a19704710058113b32",
   "language": "javascript",
   "views": {
       "by_question_id": {
           "map": "function(doc) {
                if(doc.PostTypeId === \"2\"){
                    emit(doc.ParentId, doc);
                }
                else{
                    emit(doc.Id, doc);
                }
            }"
       }
   },
   "lists": {
       "aggregated": "function(head, req){ 
                        start({\"headers\": {\"Content-Type\": \"text/json\"}});
                        var currentRow = null;
                        var currentObj = null; 
                        var retval = []; 
                        while(currentRow = getRow()){
                            if(currentObj === null || currentRow.key !== currentObj.key){
                                currentObj = {key: currentRow.key, question : null, answers : []};
                                retval.push(currentObj);
                            } 
                            if(currentRow.value.PostTypeId === \"2\"){
                                currentObj.answers.push(currentRow.value);
                            } 
                            else{
                                currentObj.question = currentRow.value;
                            }
                        }
                        send(toJSON(retval));
                    }"
   }
}

したがって、いくつかの要素をロードしたら、次のようにアクセスできます。

http://localhost:5984/<db>/_design/posts/_list/aggregated/by_question_id?<standard view limiters>

これが人々の時間を節約することを願っています。

于 2012-11-25T23:32:37.850 に答える