-
sqlserver空间数据 + c# 实现查询附近的设备
前言
一个小需求的实现,做一个备忘,个人理解,可能存在错误。
客户有很多设备,这些设备分散在不同的地方,现在需要通过小程序获取附近的(比如1000米)之类的设备列表,以距离排序
第一个想到的的是找百度/腾讯等地图,看有没有提供相应的开放api,先将我们的设备id以及对应的经纬度存储到地图中,再调用某个api,传入我当前坐标经纬度,然后返回附近的设备列表。最后没找到这个接口。应该是有,但是我没找到
第二个想能不能通过c#用啥算法实现,太菜搞不定
最后想到sqlserver有个空间数据的概念,最终勉强实现。
参考:空间数据 (SQL Server) efcore中使用空间数据
概述
空间对象其实就是点、线、面之类的意思,我们希望对其进行一些操作,如:查询得到两个面之间的交集/差集/并集;以一个点为中心,设置半径得到一个面;查询两个点之间的距离;等等。这些操作如果我们自己用算法来实现想想有点怕。
c#提供了相应的库来表示这些空间对象,也提供了相应的方法来执行针对空间对象的操作 ->NetTopologySuite(从java的JTS移植来的)由于这次小需求只是依赖了数据库对空间数据的支持,所以没有详细研究NetTopologySuite(资料少)
sqlserver2008开始支持空间数据,它也提供了相应的类型来表示空间对象,也提供相应的函数来操作这些对象,当然其它数据也支持
估计还有单独的空间数据引擎,以文档形式存储空间对象
这些点/线/面可以放在一个普通平面中,这时把他们看成几何对象,比如一个点就可以用XY坐标来表示、一个线就可以用两个点来表示...;
也可以将这些空间对象放在地球环境中来看,那么点就对象就用经纬度来表示、线就用两个点来表示
几何图形与地理位置
同一个空间对象,比如一个点 可以把它放在一个普通的平面坐标中来看待,用XY坐标来表示,也可以放进地球环境中用经纬度来表示。
- Geometry 类型表示欧几里得(平面)坐标系中的数据。
- Geography 类型表示圆形地球坐标系中的数据。
以点来说,可以创建Geometry类型的点,也可以创建Geography类型的点。数据库和c#都有对象的创建方式。同理 线、面 都分为这两种类型
空间数据类型
就是说的上面的点、线、面 当然还有更多类型。NetTopologySuite库提供了相应的类型来表示。sqlserver中也有相应的类型,但是sqlserver存储空间对象的字段的类型只分为Geography和Geometry,只是在插入值的时候会创建不同的对象
空间引用标识符 (SRID)
文档的说明看不太懂,反正对我们的限制就是 若我们使用Geography类型的空间对象,那么必须为每个空间对象设置SRID,对多个空间进行操作(比如看距离)时,这些空间对象的SRID必须一样。
感觉就像是有多个地球,每个地球有一个编号。同一个点(经纬度)在不同的地球(SRID)中其实位置不一样。不晓得这样理解对不。
在当前需求中反正写死4326,文档中经常用这个。
各种函数
实例方法:一个空间对象就是一个实例。在sql语句中可以实例化一个空间对象,数据表那个存储空间对象的单元格中存储的就是一个具体的空间对象实例
静态方法:不属于空间对象实例的,通过字符串创建空间对象,这个方法放实例上不合适
其实跟c#中的实例和静态一个意思
OGC:应该是个什么标准,定义了空间数据的通用操作,当然分为实例OGC方法和静态OGC方法
微软对标准进行了扩展,实例和静态都有扩展
原始SQL和EFCore实现
efcore不熟 又是第一次用空间数据,所以没用efcore来实现,本身用的abp 默认是用的efcore,用这个方式本来应该更方便。efcore中使用空间数据
原始sql就简单了。
没有用文档中的空间索引。也没有直接存储空间对象,而是查询时构建