使用地理数据类型时查询矩形区域(DbGeography)

时间:2014-10-11 09:19:50

标签: sql-server entity-framework google-maps spatial

我在SQL Server中查询表以在Google Maps上显示结果,我在服务器应用程序上使用Entity Framework和DbGeography数据类型。

简单地说,我从Google Maps实例获取地图边界,使用以下内容构建DbGeography实例:

var wkt = string.Format("POLYGON(({0} {1}, {0} {2}, {3} {2}, {3} {1}, {0} {1}))", EastLng, NorthLat, SouthLat, WestLng);
var bounds = DbGeography.FromText(wkt);

然后使用此查询数据库(作为更大查询的一部分):

dbset.Where(i => bounds.Intersects(l.GeographicLocation));

这很好用,我实际上已经在生产中工作了一段时间。今天我注意到有时候某些结果在地图上没有正确显示。

经过一番调查,我意识到这是因为地球的曲率; Google地图使用地球的矩形投影,但是当我创建矩形多边形时,它意味着这样的形状(从SQL Server Management Studio复制,来自正在运行的查询):

Boundary projection

查看非常大区域的地图(缩小)时效果会增加。但是当我放大时,地球曲率的影响会减小,查询似乎也能正常工作。

我需要做的是基本上在DB中查询位于Lat / Lng边界内的所有点。有没有办法做到这一点?


编辑:

这是我用来创建DbGeography实例的完整代码:

public class LatLngBounds
{
    // ...

    public DbGeography ToDbGeography()
    {
        return DbGeographyUtil.CreatePolygon(string.Format("POLYGON(({0} {1}, {0} {2}, {3} {2}, {3} {1}, {0} {1}))", EastLng, NorthLat, SouthLat, WestLng));
    }
}


public static class DbGeographyUtil
{
    public static DbGeography CreatePolygon(string wktString)
    {
        if (string.IsNullOrWhiteSpace(wktString))
            return null;

        var sqlGeography = SqlGeography.STGeomFromText(new SqlChars(wktString), DbGeography.DefaultCoordinateSystemId).MakeValid();

        var invertedSqlGeography = sqlGeography.ReorientObject();
        if (sqlGeography.STArea() > invertedSqlGeography.STArea())
        {
            sqlGeography = invertedSqlGeography;
        }

        return DbSpatialServices.Default.GeographyFromProviderValue(sqlGeography);
    }

}

编辑2:

以上示例中Google地图的坐标为:

  • 东北:纬度= 53.6 Lng = 92.5
  • 西南:Lat = 35.0 Lng = 21.2

这是我使用SQL Server Profiler捕获的正在运行的查询的一部分:

declare @p4 sys.geography
set @p4=convert(sys.geography,0xE6100000010405000000463E933FAFD64A40070000006E3F354061E412079A894140070000006E3F354061E412079A894140020000808B245740463E933FAFD64A40020000808B245740463E933FAFD64A40070000006E3F354001000000020000000001000000FFFFFFFF0000000003)

SELECT @p4;

(我添加了SELECT以查看其外观,并生成上图)

1 个答案:

答案 0 :(得分:1)

我设法通过使用DbGeometry而不是DbGeography来解决此问题

var boundsAsGeometry = DbGeometry.FromText(bounds.AsText(), bounds.CoordinateSystemId);

dbset.Where(x => boundsAsGeometry.Intersects(DbGeometry.FromText(x.GeographicLocation.AsText(), x.GeographicLocation.CoordinateSystemId)));

请注意,如果有的话,这将不会将现有的db索引用于GeographicLocation字段。