6

给定以下对象结构:

{
   key1: "...",
   key2: "...",
   data: "..."
}

有没有办法通过查询 key1 和 key2 从 CouchDB 获取此对象,而无需设置两个不同的视图(每个键一个),例如:

select * from ... where key1=123 or key2=123

亲切的问候, Artjom

编辑:

这里有一个更好的问题描述: 上面描述的对象是一个序列化的游戏状态。一个游戏只有一个创作者用户(key1)和他的对手(key2)。对于给定的用户,我想获得他参与的所有游戏(作为创造者和对手)。

4

4 回答 4

4

发出两个键(如果相等,则只发出一个):

function(doc) {
  if (doc.hasOwnProperty('key1')) {
    emit(doc.key1, 1);
  }
  if (doc.hasOwnProperty('key2') && doc.key1 !== doc.key2) {
    emit(doc.key2, 1);
  }
}

使用(正确的 url 编码)查询:

?include_docs=true&key=123

或具有多个值:

?include_docs=true&keys=[123,567,...]

更新:更新为使用单个查询查询多个值。

于 2012-05-31T05:40:21.267 回答
2

您可以创建一个 CouchDB 视图,该视图产生如下输出:

["key1", 111],
["key1", 123],
["key2", 111],
["key2", 123],
etc.

用 javascript 编写地图视图非常简单:

function(doc) {
    emit(["key1", doc["key1"]], null);
    emit(["key2", doc["key2"]], null);
}

查询时,可以使用多个key进行查询:

{"keys": [["key1", 123], ["key2", 123]]}

您可以将该 JSON 作为 POST 中的数据发送到视图。或者最好为您的编程语言使用 API。此查询的结果将是视图中与任一键匹配的每一行。因此,在 key1 和 key2 上匹配的每个文档都将在视图结果中返回两行。

于 2012-05-30T16:47:28.323 回答
1

我也在努力解决类似的问题,如何使用

"select * from ... where key1=123 or key2=123". 

以下视图将允许您按姓氏名字字段查找客户文档:

function(doc) {
  if (doc.Type == "customer") {
      emit(doc.LastName, {FirstName: doc.FirstName, Address: doc.Address});
      emit(doc.FirstName, {LastName: doc.LastName, Address: doc.Address});
  }
}
于 2017-01-08T12:54:02.687 回答
0

我将它用于查询我所有文档并返回与节点的存在和查询匹配的每个文档的 Web 服务。在此示例中,我使用节点“详细信息”进行搜索。如果您想搜索不同的节点,则需要指定。

这是我的第一篇 Stack Overflow 帖子,所以我希望我能帮助别人:)

***Python Code

import tornado.httpserver
import tornado.ioloop
import tornado.options
import tornado.web
import httplib, json

from tornado.options import define,options

define("port", default=8000, help="run on the given port", type=int)


class MainHandler(tornado.web.RequestHandler):
    def get(self):
        db_host = 'YOUR_COUCHDB_SERVER'
        db_port = 5984
        db_name = 'YOUR_COUCHDB_DATABASE'

        node = self.get_argument('node',None)
        query = self.get_argument('query',None)

        cleared = None
        cleared = 1 if node else self.write('You have not supplied an object node.<br>')
        cleared = 2 if query else self.write('You have not supplied a query string.<br>')

        if cleared is 2:
            uri = ''.join(['/', db_name, '/', '_design/keysearch/_view/' + node + '/?startkey="' + query + '"&endkey="' + query + '\u9999"'])
            connection = httplib.HTTPConnection(db_host, db_port)
            headers = {"Accept": "application/json"}
            connection.request("GET", uri, None, headers)
            response = connection.getresponse()
            self.write(json.dumps(json.loads(response.read()), sort_keys=True, indent=4))

class Application(tornado.web.Application):
    def __init__(self):
        handlers = [
            (r"/", MainHandler)
        ]
        settings = dict(
            debug = True
        )
        tornado.web.Application.__init__(self, handlers, **settings)

def main():
    tornado.options.parse_command_line()
    http_server = tornado.httpserver.HTTPServer(Application())
    http_server.listen(options.port)
    tornado.ioloop.IOLoop.instance().start()

if __name__ == '__main__':
    main()


***CouchDB Design View
{
   "_id": "_design/keysearch",
   "language": "javascript",
   "views": {
       "detail": {
           "map": "function(doc) { var docs = doc['detail'].match(/[A-Za-z0-9]+/g); if(docs) { for(var each in docs) { emit(docs[each],doc); } } }"
       }
   }
}
于 2013-12-16T21:17:22.567 回答