C#中的MongoDB地理空间索引

4

我一直在尝试使用C#官方驱动程序创建和查询MongoDB,但总是遇到同样的问题。问题在于如何创建带有地理信息的数据。我找不到答案。

代码:

MongoUrl url = new MongoUrl("mongodb://xxx.xx.x.xx/mydb");
MongoServer server = MongoServer.Create(url);
MongoDatabase database = server.GetDatabase("mydb");

<-- 这个可以正常工作

BsonDocument[] batch = {
                         new BsonDocument {
                                             { "name", "Bran" },
                                             { "loc", "10, 10" }
                                         },
                                        new BsonDocument {
                                            { "name", "Ayla" },
                                            { "loc", "0, 0" }
                                        }
            };

places.InsertBatch(batch);

<-- 那部分有些错误

places.EnsureIndex(IndexKeys.GeoSpatial("loca"));
var queryplaces = Query.WithinCircle("loca", 0, 0, 11);
var cursor = places.Find(queryplaces);
foreach (var hit in cursor)
{
    foreach (var VARIABLE in hit)
    {
        Console.WriteLine(VARIABLE.Value);
    }
}

<-- 我认为这部分应该显示两个文档,现在却没有显示。简单的查找可以找到它们两个。 需要一些帮助。

2个回答

6

以下是C#语言的示例(需要注意数组的顺序,即经度在前,纬度在后——这遵循了更符合逻辑的x、y顺序,而不是常见的纬度在经度之前):

1.) 首先,您的类需要包含以下内容:

public double[] Location { get; set; }

public double Latitude
{
    get { return _latitude; }
    set
    {
        Location[1] = value;
        _latitude = value;
    }
}

public double Longitude
{
    get { return _longitude; }
    set
    {
        Location[0] = value;
        _longitude = value;
    }
}

public MyClass()
{
    Location = new double[2];
}

2.) 接下来是使用官方C#驱动程序并使用地理编码执行插入操作的代码示例:

    /// <summary>
    /// Inserts object and creates GeoIndex on collection (assumes TDocument is a class
    /// containing an array double[] Location where [0] is the x value (as longitude)
    /// and [1] is the y value (as latitude) - this order is important for spherical queries.
    /// 
    /// Collection name is assigned as typeof(TDocument).ToString()
    /// </summary>
    /// <param name="dbName">Your target database</param>
    /// <param name="data">The object you're storing</param>
    /// <param name="geoIndexName">The name of the location based array on which to create the geoIndex</param>
    /// <param name="indexNames">optional: a dictionary containing any additional fields on which you would like to create an index
    /// where the key is the name of the field on which you would like to create your index and the value should be either SortDirection.Ascending
    /// or SortDirection.Descending. NOTE: this should not include geo indexes! </param>
    /// <returns>void</returns>
    public static void MongoGeoInsert<TDocument>(string dbName, TDocument data, string geoIndexName, Dictionary<string, SortDirection> indexNames = null)
    {
        Connection connection = new Connection(dbName);
        MongoCollection collection = connection.GetMongoCollection<TDocument>(typeof(TDocument).Name, connection.Db);
        collection.Insert<TDocument>(data);
        /* NOTE: Latitude and Longitude MUST be wrapped in separate class or array */
        IndexKeysBuilder keys = IndexKeys.GeoSpatial(geoIndexName);
        IndexOptionsBuilder options = new IndexOptionsBuilder();
        options.SetName("idx_" + typeof(TDocument).Name);
        // since the default GeoSpatial range is -180 to 180, we don't need to set anything here, but if
        // we wanted to use something other than latitude/longitude, we could do so like this:
        // options.SetGeoSpatialRange(-180.0, 180.0);

        if (indexNames != null)
        {
            foreach (var indexName in indexNames)
            {
                if (indexName.Value == SortDirection.Decending)
                {
                    keys = keys.Descending(indexName.Key);
                }
                else if (indexName.Value == SortDirection.Ascending)
                {
                    keys = keys.Ascending(indexName.Key);
                }
            }
        }

        collection.EnsureIndex(keys, options);

        connection.Db.Server.Disconnect();
    }


using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using MongoDB.Bson;
using MongoDB.Driver;

namespace MyMongo.Helpers
{
    public class Connection
    {
        private const string DbName = "";
        private const string Prefix = "mongodb://";
        //private const string Server = "(...):27017/";
        private const string Server = "localhost:27017/";
        private const string PassWord = "";
        private const string UserName = "";
        private const string Delimeter = "";
        //if using MongoHQ
        //private const string Delimeter = ":";
        //private const string Prefix = "mongodb://";
        //private const string DbName = "(...)";
        //private const string UserName = "(...)";
        //private const string Server = "@flame.mongohq.com:(<port #>)/";
        //private const string PassWord = "(...)";
        private readonly string _connectionString = string.Empty;

        public MongoDatabase Db { get; private set; }
        public MongoCollection Collection { get; private set; }

        public Connection()
        {
            _connectionString = Prefix + UserName + Delimeter + PassWord + Server + DbName;
        }

        public Connection(string dbName)
        {
            _connectionString = Prefix + UserName + Delimeter + PassWord + Server + DbName;
            Db = GetDatabase(dbName);
        }

        //mongodb://[username:password@]hostname[:port][/[database][?options]]
        public MongoDatabase GetDatabase(string dbName)
        {
            MongoServer server = MongoServer.Create(_connectionString);
            MongoDatabase database = server.GetDatabase(dbName);
            return database;
        }

        public MongoCollection<TDocument> GetMongoCollection<TDocument>(string collectionName, MongoDatabase db, SafeMode safeMode = null)
        {
            if (safeMode == null) { safeMode = new SafeMode(true); }
            MongoCollection<TDocument> result = db.GetCollection<TDocument>(collectionName, safeMode);
            return result;
        }
    }
}

谢谢,我认为这个例子非常正确,但有点高级。 - JustusTh
您需要在每次插入数据时都添加索引吗?或者只需在第一次添加索引到Mongo集合中,每次插入数据时会自动“刷新”索引即可? - diegosasw
@iberodev 这个答案是很久以前写的 - 我不能确定它是否仍然是最新的。 - Jordan

2

在查找和搜索后,我在这里找到了答案:https://github.com/karlseguin/pots-importer/blob/master/PotsImporter/NodeImporter.cs

阅读我的第一段代码时,请注意这个链接可以解决问题。

  MongoCollection<BsonDocument> places =
               database.GetCollection<BsonDocument>("places");

            BsonDocument[] batch = {
                                       new BsonDocument { { "name", "Bran" }, { "loc", new BsonArray(new[] { 10, 10 }) } },
                                       new BsonDocument { { "name", "Ayla" }, { "loc", new BsonArray(new[] { 0, 0 }) } }
            };

            places.InsertBatch(batch);

            places.EnsureIndex(IndexKeys.GeoSpatial("loc"));

            var queryplaces = Query.WithinCircle("loc", 5, 5, 10);
            var cursor = places.Find(queryplaces);
            foreach (var hit in cursor)
            {
                Console.WriteLine("in circle");
                foreach (var VARIABLE in hit)
                {
                    Console.WriteLine(VARIABLE.Value);

                }
            }

澄清一下:问题代码的问题在于位置信息不应该被存储为字符串,而是应该作为一个包含两个元素(x,y)的数组。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接