因此,根据 Phil 的建议,我使用空间重写了很多内容。效果很好,您只需要 .NET4.5、EF5+ 和 sql server(我相信是 2008 年及以上版本)。我正在使用 EF6 + Sql Server 2012。
Set-Up
第一步是添加一个Geography
列到我的数据库EventListings
表(右键单击它 - >设计)。我将我的位置命名为:
接下来,由于我使用数据库优先的 EDM,因此我必须更新模型以使用我创建的新字段。然后我收到一个关于无法将 Geography 转换为 double 的错误,因此我修复它的方法是选择实体中的 Location 属性,转到其属性并将其类型从 double 更改为Geography
:
在半径范围内查询并添加带有位置的事件
然后你所要做的就是像这样查询你的实体集合:
var events = EventRepository.EventListings
.Where(x => x.Location.Distance(originCoordinates) * 0.00062 <= radiusParam);
Distance 扩展方法获取从当前对象到您传入的“其他”对象的距离。该其他对象的类型为DbGeography
。您只需调用一个静态方法,它就会创建其中一只小狗,然后将您的经度和纬度作为一个点放入其中:
DBGeography originCoordinates = DBGeography.fromText("Point(" + originLongitude + " " + originLatitude + ")");
这不是我创建 originCoords 的方式。我下载了一个单独的数据库,其中包含所有邮政编码及其纬度和经度的列表。我添加了一列类型Geography
对此也是如此。我将在本答案的最后展示如何操作。然后,我查询邮政编码上下文,以从 ZipCode 表中的 Location 字段获取 DbGeography 对象。
如果用户想要一个比邮政编码更具体的来源,我会调用 Google Maps API(更具体的是 GeoCode)并通过 Web 服务获取用户特定地址的纬度和经度,并从创建一个 DBGeography 对象响应中的纬度和经度值。
我在创建活动时也使用 google API。在将我的实体添加到 EF 之前,我只是像这样设置了位置变量:
someEventEntity.Location = DBGeography.fromText("Point(" + longitudeFromGoogle+ " " + latitudeFromGoogle + ")");
其他提示和技巧以及故障排除
我如何获得距离延伸法?
获取扩展方法Distance
您必须添加对您的项目的引用:System.Data.Entity
执行此操作后,您必须添加使用:using System.Data.Entity.Spatial;
到你的班级。
The Distance
扩展方法返回距离米的计量单位(我认为你可以更改它,但这是默认的)。在这里,我的半径参数以英里为单位,因此我做了一些数学转换。
Note: 有一个DBGeography
班级在System.Data.Spatial
。这是错误的这对我不起作用。我在网上找到的很多例子都使用了这个。
如何将纬度/经度转换为地理列
所以,如果你像我一样下载了一个包含所有信息的邮政编码数据库Latitude
& Longitude
列,然后意识到它没有地理列......这可能会有所帮助:
1) 在表中添加位置列。将类型设置为地理。
2)执行以下sql
UPDATE [ZipCodes]
SET Location = geography::STPointFromText('POINT(' + CAST([Longitude] AS VARCHAR(20)) + ' ' + CAST([Latitude] AS VARCHAR(20)) + ')', 4326)
如何查看地理项目的详细信息
因此,当您插入一些 DbGeography 项目后在 Sql Server Management Studio 中查询 EventListings 表时,您将看到Location
列包含一个十六进制值,例如:0x1234513462346。当您想确保插入正确的值时,这不是很有帮助。
要实际查看该字段的纬度和经度,您必须像这样查询它:
SELECT Location.Lat Latitude, Location.Long Longitude
FROM [EventListings]