利用MySQL空间函数实现位置打卡的完整步骤
前言
项目需求是跟用户当前位置判断是否在给定的地理位置范围内,符合位置限制才可以打卡,其中的位置范围是一个或多个不规则的多边形。如下图,判断用户是在清华还是北大。
图形获取区域坐标#
因为项目前端使用微信小程序的wx.getLocation获取地理位置,为了坐标的一致性,后台选取区域范围采用了腾讯地图的地理位置服务,在应用工具->绘制几何图形里,提供了点、线、多边形和圆形可以方便的选取看这里。
在官方提供的示例上稍加改动即可获取选定的位置坐标。
存储位置
取到坐标位置后,接着就是怎么存储?
开放地理空间联盟(OGC)是一个由250多家公司,机构和大学组成的国际联盟,参与开发公开可用的空间解决方案,这些解决方案可用于管理空间数据的各种应用程序。OGC发布了地理信息的OpenGIS®Implementation标准,该规范可从OGC网站http://www.opengeospatial.org/standards/sfs获得。为了遵循OGC规范,MySQL将空间extensions实现为具有GeometryTypes环境的SQL的子集,提供生成、存储、分析空间的功能。总之,MySQL可以满足我们的需求。
MySQL提供单个的存储类型POINT、LINESTRING、POLYGON对应几何图形点、线、多边形,GEOMETRY可以存储三种中的任何一种。同时拥有存储多种类型的能力,MULTIPOINT、MULTILINESTRING、MULTIPOLYGON、GEOMETRYCOLLECTION依次对应单个图形的复数。
回到项目中,我们用到的是POLYGON,
建表语句如下:
CREATETABLE`polygon`( `id`int(10)unsignedNOTNULLAUTO_INCREMENT, `name`varchar(255)DEFAULTNULL, `polygon`polygonNOTNULL, PRIMARYKEY(`id`), SPATIALKEY`d`(`polygon`) )DEFAULTCHARSET=utf8;
插入数据
MySQL支持将Well-Known文本(WKT)格式和Well-Known二进制(WKB)格式两种格式转换为object类型存储起来,我们使用更易于理解的WKT格式。对WKB感兴趣的可以看这里。
插入语句如下:
INSERTINTO`polygon`VALUES('1','清华大学',GeomFromText('POLYGON(( 40.01169924229143116.31565081888039,39.99304082299905116.31616541796757,39.99343506780591116.33297565023167,40.00237067000859116.33743550702275,40.01340715321479116.33057418815224,40.01169924229143116.31565081888039))')); INSERTINTO`polygon`VALUES('2','北京大学',GeomFromText('POLYGON((39.99711457525893116.30450117461078,39.98673259872773116.30535884106575,39.98673259872773116.31702308311287,39.99963848242885116.31598375134854,39.99711457525893116.30450117461078))'));
需要注意的是腾讯地图返回的多边形的点不是闭合的,而polygon函数需要为了确定多边形是否闭合要求第一个点和最后一个点是一样的。如果不是闭合的polygon返回的结果将是NULL,插入语句就会执行失败。
如果几何满足诸如此(非穷举)列表中的条件,则它在语法上是well-formed:
- 线串至少有两个点
- 多边形至少有一个环
- 多边形环关闭(第一个和最后一个点相同)
- 多边形环至少有4个点(最小多边形是一个三角形,第一个和最后一个点相同)
- 集合不为空(除了GeometryCollection)
查询判断
SELECT*FROMpolygonWHERE MBRWithin(ST_GeomFromText('POINT(39.991333490218544116.30964748487895)'),polygon); #在北京大学 SELECT*FROMpolygonWHERE MBRWithin(ST_GeomFromText('POINT(39.988967560246685116.3286905102832)'),polygon); #不在北大
细心的同学可能发现了这里的查询语句里用的是函数,在以往的SQL里如果存在查询字段上使用函数必然导致索引失效、全表扫描,但是在空间数据上不会,先看EXPLAIN语句和结果:
可见MySQL空间类型的数据同样可以建立索引,使用的关键词是SPATIAL
用法如下:
CREATETABLEgeom(gGEOMETRYNOTNULL); CREATESPATIALINDEXgONgeom(g);
常用的空间计算函数
1、判断两点之间的距离
ST_Distance(g1,g2),返回g1和g2之间的距离。如果任一参数是NULL或空几何,则returnvalue为NULL。
2、图形1是否完全包含图形2
ST_Contains(g1,g2),返回1或0以指示g1是否完全包含g2。还可以用ST_Within(g2,g1)达到相同的效果。
3、不相交
ST_Disjoint(g1,g2),返回1或0以指示g1是否在空间上与(不相交)g2不相交。
4、关于图形相交的情况比较复杂,包含重叠、外相交等情况,具体可以看这里
总结
本文通过一个地理位置打卡的需求,使用MySQL自带的Polygon数据类型实现了空间数据的存储,用ST_Contains(g1,g2)函数代入了后台预置的地理区域和前端获取到的用户地理位置可以得出用户是否在打卡范围内。其中还涉及到了MySQL在使用函数作为查询字段的情况下依然可以使用索引,最后延伸了一些其他的空间处理函数。
到此这篇关于利用MySQL空间函数实现位置打卡的文章就介绍到这了,更多相关MySQL空间函数位置打卡内容请搜索毛票票以前的文章或继续浏览下面的相关文章希望大家以后多多支持毛票票!
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。