本文使用官方C# Driver,实现在MongoDB中存储,查询空间数据(矢量)
//打开MongoDB的Collection MongoDatabase db = server.GetDatabase("aa"); MongoCollection colSheng = db.GetCollection("sheng"); //使用Ogr库打开Shapefile文件 DataSource ds = Ogr.Open(@"c:\temp\sheng.shp", 0); Layer lyr = ds.GetLayerByIndex(0); //读取要素数量和字段数量 int feaCount = lyr.GetFeatureCount(0); int fieldCount = lyr.GetLayerDefn().GetFieldCount(); //读取所有字段名 List<string> fieldNames =new List<string>(); for (int i = 0; i < fieldCount; i++) { fieldNames.Add(lyr.GetLayerDefn().GetFieldDefn(i).GetName()); } //循环将所有要素添加到MongoDB中 for (int i = 0; i < feaCount; i++) { //使用Ogr库将矢量要素的空间信息转成Json格式 Feature fea = lyr.GetFeature(i); Geometry geo = fea.GetGeometryRef(); string json = geo.ExportToJson(null); BsonDocument doc = new BsonDocument(); //将Json格式的空间信息存到Collection中 //BsonValue bs = BsonValue.Create(json); //这种方法是不可以的,添加到库里之后无法使用空间查询语句查询 BsonValue bs2 = BsonDocument.Parse(json); //这种方法才是正确的 //doc.Add(new BsonElement("geom", bs)); doc.Add(new BsonElement("geo",bs2)); //通过循环将所有字段的属性信息存入Collection中 for (int j = 0; j < fieldCount; j++) { string tmpFieldVal = fea.GetFieldAsString(j); doc.Add(new BsonElement(fieldNames[j],tmpFieldVal)); } var res = colSheng.Insert<BsonDocument>(doc); }
> db.sheng.find().limit(1)
{ "_id" : ObjectId("5371bf4e1dbba31914224563"), "geo" : { "type" : "Polygon", "coordinates" : [ [ [ 89.8496, 14.093 ], [ 90.3933, 14.004 ], [ 90.2708, 13.4708 ], [ 89.7284, 13.5597 ], [ 89.8496, 14.093 ] ] ] }, "pyname" : "sx", "boxtype" : "inter", "date" : "2012/6/5 12:41:42" }
//获取Collection MongoDatabase db = server.GetDatabase("aa"); MongoCollection colSheng = db.GetCollection("sheng"); //定义一个查询框或查询多边形 var poly = GeoJson.Polygon<GeoJson2DCoordinates>( GeoJson.Position(100, 20), GeoJson.Position(110, 20), GeoJson.Position(110, 40), GeoJson.Position(100, 40), GeoJson.Position(100, 20)); //以这个查询多边形为条件定义一条查询语句 var queryFilter2 = Query.GeoIntersects("geo", poly); //进行查询,输出MongoCursor cur = colSheng.FindAs<BsonDocument>(queryFilter2).SetFields( "pyname", "date"); //获取结果 var res = cur.ToArray(); for (int i = 0; i < res.Count(); i++) { BsonDocument tmpDoc = res.ElementAt(i); //do something you want }
> db.test.find() { "_id" : ObjectId("535884771dbba31858ad2101"), "geo" : { "type" : "Polygon", "coordinates" : [ [ [ 96.722, 38.755 ], [ 97.3482, 38.6922 ], [ 97.1674, 38.0752 ], [ 96.5474, 38.1383 ], [ 96.722, 38.755 ] ] ] } }
> db.test.find({ "geo" : { "$geoIntersects" : { "$geometry" : { "type" : "Polygon", "coordinates" : [[[91.0, 33.0], [102.0, 33.0], [102.0, 38.0], [91.0, 38.0], [91.0, 33.0]]] } } } })
{ "_id" : ObjectId("535884771dbba31858ad2101"), "geo" : { "type" : "Polygon", "coordinates" : [ [ [ 96.722, 38.755 ], [ 97.3482, 38.6922 ], [ 97.1674, 38.0752 ], [ 96.5474, 38.1383 ], [ 96.722, 38.755 ] ] ] } }
//获取Collection MongoDatabase db = server.GetDatabase("aa"); MongoCollection colsheng= db.GetCollection("sheng"); //查询所有记录 var cur = colsheng.FindAllAs<BsonDocument>(); long totalCount = cur.Count(); //遍历所有记录 for (int i = 0; i <= totalCount/1000; i++) { if (i * 1000 >= totalCount) continue; int skip = i * 1000; var cur2 = cur.Clone<BsonDocument>().SetSkip(skip).SetLimit(1000); var lst = cur2.ToArray(); for (int j = 0; j < lst.Count(); j++) { //获取一条记录对应的BsonDocument BsonDocument doc = lst[j]; var id = doc["_id"]; //该记录对应的ID BsonDocument geo = doc["geo"].ToBsonDocument(); string geostr = geo[1].ToString(); //该记录对应空间信息的Json字符串 List<double> coords = GetCoordLstFromString(geostr); //解析Json串,获得所有点的坐标(这里子函数就省略了) double xmin = 181, xmax = -181, ymin = 91, ymax = -91; //四个边界值,由于图层为经纬度,所以初值设为这些值 //计算最大最小值 for (int k = 0; k < coords.Count; k++) { if (k % 2 == 0) { if (coords[k] < xmin) xmin = coords[k]; if (coords[k] > xmax) xmax = coords[k]; } else { if (coords[k] < ymin) ymin = coords[k]; if (coords[k] > ymax) ymax = coords[k]; } } //将最大最小值写入Collection var tmpQuery = Query.EQ("_id", id); var tmpUpdate = MongoDB.Driver.Builders.Update.Set("xmax", xmax); var tmpres = col02c.Update(tmpQuery, tmpUpdate); tmpUpdate = MongoDB.Driver.Builders.Update.Set("xmin", xmin); tmpres = col02c.Update(tmpQuery, tmpUpdate); tmpUpdate = MongoDB.Driver.Builders.Update.Set("ymax", ymax); tmpres = col02c.Update(tmpQuery, tmpUpdate); tmpUpdate = MongoDB.Driver.Builders.Update.Set("ymin", ymin); tmpres = col02c.Update(tmpQuery, tmpUpdate); } }
//获取Collection MongoDatabase db = server.GetDatabase("aa"); MongoCollection colSheng = db.GetCollection("zy02c"); //第一步,通过四边界筛选, var query = Query.And(Query.GT("xmax", 91.0), Query.LT("xmin", 102.0), Query.GT("ymax", 33.0), Query.LT("ymin", 38.0)); var cur = colSheng.FindAs<BsonDocument>(query); //定义第二空间运算时的条件多边形(Ogr格式的定义) Geometry queryGeoLR = new Geometry(wkbGeometryType.wkbLinearRing); queryGeoLR.AddPoint(91.0, 33.0,0); queryGeoLR.AddPoint(102.0, 33.0,0); queryGeoLR.AddPoint(102.0, 38.0,0); queryGeoLR.AddPoint(91.0, 38.0,0); queryGeoLR.AddPoint(91.0, 33.0,0); Geometry queryGeo = new Geometry(wkbGeometryType.wkbPolygon); queryGeo.AddGeometry(queryGeoLR); //循环查询到的结果 var lst = cur.ToArray(); for (int i = lst.Length-1; i >=0; i--) { //获取当前记录对应的BsonDocument BsonDocument doc = lst[i]; var id = doc["_id"]; //当前记录的ID BsonDocument geo = doc["geo"].ToBsonDocument(); string geostr = geo[1].ToString(); //当前记录对应空间信息的Json字符串 //通过Json串获取坐标值,并生成对应的Geometry对象 List<double> coords = GetCoordLstFromString(geostr); Geometry resGeoLR = new Geometry(wkbGeometryType.wkbLinearRing); for (int j = 0; j < coords.Count / 2; j++) { resGeoLR.AddPoint_2D(coords[j], coords[j + 1]); } resGeoLR.AddPoint_2D(coords[0], coords[1]); Geometry resGeo = new Geometry(wkbGeometryType.wkbPolygon); resGeo.AddGeometry(resGeoLR); //判断是该Geometry与条件多边形是否相交 if (resGeo.Intersects(queryGeo)) { //do something } }
MongoDB中空间数据的存储和操作,布布扣,bubuko.com
原文:http://blog.csdn.net/rrrrssss00/article/details/25784263