LBS解决方案

LBS解决方案
  • LBS(基于地理位置的服务)服务是现在移动互联网中比较常用的功能,例如外卖中我附近的店铺,通常是以客户位置坐标为中心,查询一定范围内的店铺信息,按照距离由近及原进行倒叙排序
方案一,直接mysql
  • 经纬度范围计算就是弧度的计算,只要数学够好,算出来分分钟的事情,直接上mysql一个语句搞定:
CREATE TABLE `places` (`id` INT (11) NOT NULL AUTO_INCREMENT,`lat` DOUBLE NOT NULL DEFAULT '0',`lng` DOUBLE NOT NULL DEFAULT '0',PRIMARY KEY (`Id`),KEY `lat` (`lat`),KEY `lng` (`lng`)
) ENGINE = INNODB AUTO_INCREMENT = 9 DEFAULT CHARSET = utf8;-- 测试经纬度
INSERT INTO `zhenai_yewu`.`places` (`lat`, `lng`) VALUES ( 23.123123, 123.123);
INSERT INTO `zhenai_yewu`.`places` (`lat`, `lng`) VALUES ( 21.123123, 121.123);
INSERT INTO `zhenai_yewu`.`places` (`lat`, `lng`) VALUES ( 22.123123, 122.123);
INSERT INTO `zhenai_yewu`.`places` (`lat`, `lng`) VALUES ( 24.123123, 124.123);
INSERT INTO `zhenai_yewu`.`places` (`lat`, `lng`) VALUES ( 25.123123, 125.123);
INSERT INTO `zhenai_yewu`.`places` (`lat`, `lng`) VALUES ( 26.123123, 126.123);
INSERT INTO `zhenai_yewu`.`places` (`lat`, `lng`) VALUES ( 27.123123, 127.123);
INSERT INTO `zhenai_yewu`.`places` (`lat`, `lng`) VALUES ( 28.123123, 128.123);-- 为了查询效果,将范围设置为255555 单位是KM
SELECT id, ( 6371 * acos( cos( radians(24.123123) ) * cos( radians( lat ) ) * cos( radians
( lng ) - radians(124.123) ) + sin( radians(24.123123) ) * sin( radians( lat ) ) ) ) AS distanceFROM places HAVING distance < 255555 ORDER BY distance LIMIT 0 , 100;
-- 结果信息 id: distance
12	0
13	150.27231636470339
9	150.80702908932616
14	299.9866451782974
11	302.12576883654754
15	449.11942176001054
10	453.93347262147796
16	597.6467747050862
  • sql 查询的是lat,lng对应的坐标信息,中心点的信息是[124.123, 24.123123],明显这个是通过弧度计算来算出适合范围内的经纬度范围,然后搜索对应的id,如下sql语句分析
    在这里插入图片描述
  • 即使加上了索引这种查询也必须扫全表信息,效率低下
  • 优化一下:
    • 另外一种算法,可以根据圆心坐标计算正方形四个点的坐标,然后查询正方形内的点。
SELECT * FROM places WHERE ((lat BETWEEN 20 AND 29) AND (lng BETWEEN 90 AND 170));
//获取如下结果:
9	23.123123	123.123
10	21.123123	121.123
11	22.123123	122.123
12	24.123123	124.123
13	25.123123	125.123
14	26.123123	126.123
15	27.123123	127.123
16	28.123123	128.123
  • 这样优化后,虽然数据不完全精确(圆形变正方形),但性能提升很明显,并且可以通过给lat lng字段做索引的方式进一步加快这条SQL的查询速度。对精度有要求的应用也可以在这个结果上再进行计算,排除那些在方块范围内但不在圆形范围内的数据,已达到对精度的要求。 但是没有排序,没有距离,除非做其他运算。
方案二,Redis
  • Redis3.2.0后开始支持LBS相关的命令,要实现以上功能,主要用到Redis geo相关的两个命令 GEOADD, GETRADIOUS
  • 命令描述,案例:
GEOADD key longitude latitude member [longitude latitude member ...]
//命令将指定地理空间位置(金纬度信息,名称)添加到指定key中
  • 有效经度值域[-180,180], 有效纬度值域[-85.05112878 , 85.05112878 ],单坐标超过上述指定范围时候,会返回错误信息,可以一次添加多个位置地点,如下案例:
新docker-redis:0>GEOADD location 117 20 beijing
"1"
新docker-redis:0>GEOADD location 120 30 shenzheng
"1"
新docker-redis:0>GEOADD location 121 31 guangzhou, 122 32 nanjing
"2"
  • 添加后通过GEORADIOUS查询如下:
GEORADIUS key longitude latitude radius m|km|ft|mi [WITHCOORD] [WITHDIST] [WITHHASH] [COUNT count]
  • 命令以给定的经纬度为中心,返回键值key包含的位置元素中,与中心距离不超过给定最大距离的所有位置的元素,并且可以指定单位,顺序返回单位信息如下:
m:米
km:千米
mi:英里
ft:英尺
  • 给定一下选项时候,命令会额外返回信息:
  • WITHDIST:返回位置元素的同时,将位置元素与中心之间的距离也一并返回。距离的单位和用户给定的范围单位保持一致 如下:

新docker-redis:0>GEORADIUS location 120.987 30.1234 500 km WITHDIST1)    1)   "shenzheng"2)   "95.9992"2)    1)   "guangzhou,"2)   "97.5090"3)    1)   "nanjing"2)   "229.9580"
  • WITHCOORD :将位置元素的进度和纬度也一起返回。

新docker-redis:0>GEORADIUS location 120.987 30.1234 500 km WITHCOORD1)    1)   "shenzheng"2)      1)    "120.00000089406967163"2)    "30.00000024997701331"2)    1)   "guangzhou,"2)      1)    "120.99999815225601196"2)    "31.00000097648057817"3)    1)   "nanjing"2)      1)    "122.00000077486038208"2)    "31.99999916826298119"
  • WITHHASH 以52位有符号整形,返回位置元素经过原始geohash编码的有序集合分值
新docker-redis:0>GEORADIUS location 120.987 30.1234 500 km WITHHASH1)    1)   "shenzheng"2)   "4054115787372083"2)    1)   "guangzhou,"2)   "4054742425916764"3)    1)   "nanjing"2)   "4066587848444692"
  • asc:更具中心位置按从近到远的方式返回位置元素
  • desc:根据中心位置,从远到近的方式返回位置元素
  • 默认情况GEORADIUS命令会返回所有匹配的位置元素,用户可以使用count 获取前面N个
新docker-redis:0>GEORADIUS location 120.987 30.1234 500 km WITHHASH asc1)    1)   "shenzheng"2)   "4054115787372083"2)    1)   "guangzhou,"2)   "4054742425916764"3)    1)   "nanjing"2)   "4066587848444692"新docker-redis:0>GEORADIUS location 120.987 30.1234 500 km WITHHASH desc1)    1)   "nanjing"2)   "4066587848444692"2)    1)   "guangzhou,"2)   "4054742425916764"3)    1)   "shenzheng"2)   "4054115787372083"新docker-redis:0>GEORADIUS location 120.987 30.1234 500 km WITHHASH desc count 11)    1)   "nanjing"2)   "4066587848444692"
方案三
  • mongodb 原生支持地理位置索引2dSpace,可以直接用于位置距离计算和查询。 而且我们用的也比较多,能够支持地理位置查询的同事也有免息集合存储,数据格式自由,查询性能高等特点,对应我们需求只需要一个mongodb查询语句:
 db。moment.find( { location : { $geoWithin :{ $centerSphere :[ [121.492183, 31.247610 ] , 1000 / 3963.2 ]} } } )
  • 查询结果默认由近及远,并且geoWithin是mongodb支持的查询函数,已经在性能上做了高度的优化,完全可以生产用,现在moment的地理位置推荐用的这个
方案四
  • GeoHash是一种地址编码,通过切分地图区域变成小的方块,只有切分的足够小,精度就越高。

上一篇:Redis分布式锁奥义
下一篇:Redis遍历方式思考–字典扩容方式

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/310574.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

长沙IT技术圈百万年薪大佬?是否存在?

作者&#xff1a;邹溪源&#xff0c;长沙资深互联网从业者&#xff0c;架构师社区特邀嘉宾&#xff01;01引子不知不觉&#xff0c;IT技术圈开始流传起“百万年薪”的故事&#xff0c;有人问我&#xff0c;长沙有百万大佬么&#xff1f;其实我也不知道。02背景长沙自古以来就是…

网络编程-网络分层的意义

网络 我们生活在一个网络无处不在的一个虚拟世界中&#xff0c;网络中的每一个设备都是一个节点。大多是我们的计算机&#xff0c;但是他还可以连接其他设备&#xff0c;例如打印机&#xff0c;路由器&#xff0c;网关&#xff0c;你的手机&#xff0c;智能家居等。我们可以使…

网络编程-TCP/IP协议栈-IP协议

协议 协议就是约定的一种规则&#xff0c;例如扑克游戏中约定好的各种规则&#xff0c;2<3<4<5<…等&#xff0c;以此作为游戏规则。当所有人都遵循这个规则&#xff0c;那么久可以不需要任何多余的交流就可以进行游戏&#xff0c;这个方式形成的约定就是一种协议…

.NET项目升级手记:可为空引用

c# 8引入了新特性&#xff1a;“可为空引用”&#xff08;详情&#xff09;&#xff0c;这个功能个人觉得挺好的&#xff0c;能够非常明确的表现程序设计者的意图&#xff0c;编译器能够进行检查&#xff0c;尽最大可能减小NullReferenceException错误。如果是新项目&#xff0…

.NET与鲲鹏共展翅,昇腾九万里(一)

2019年1月7日&#xff0c;华为推出鲲鹏920处理器&#xff0c;便宣告了构建鲲鹏生态系统的开始。据官方介绍&#xff0c;鲲鹏是一个包含了鲲鹏计算单元、AI处理单元、智能管理、智能网卡的片上系统SoC&#xff0c;在此之上加上服务器操作系统&#xff0c;从而形成一个生态的闭环…

网络编程-TCP/IP协议栈-TCP协议

TCP协议 TCP协议作用 TCP协议位于协议栈的传输层。当应用层向TCP层发送用于网间传输的&#xff0c;用8字节表示的数据流&#xff0c;TCP则吧数据流分割成适当长度的报文段&#xff0c;最大传输段大小&#xff08;MSS&#xff09;通常受到改计算机连接的网络数据链路层的最大传…

互联网10年,激战如梦

— 1 —1969年&#xff0c;美国国防部研究计划署第一次将互联网应用于军事连接。随后美国西南部四所名校的四台计算机通过这项技术连接起来。谁也没想到&#xff0c;这项计划会对人类的命运产生如此重大影响。1993年&#xff0c;互联网真正诞生。美国白宫宣布开始提供「在线服务…

网络编程-TCP/IP协议栈-UDP/HTTP协议

UDP协议 UDP协议全称是用户数据报协议&#xff0c;在网络中她与TCP协议一样用于处理数据包&#xff0c;两个协议同处于协议栈的传输层&#xff0c;和TCP不同的是&#xff0c;UDP是一种无连接的协议栈。 因为UDP是无连接的&#xff0c;所以相对来说&#xff0c;UDP的报头比TCP要…

十问十答 Ms-PL 许可证

Microsoft 公共许可证&#xff08;The Microsoft Public License&#xff09;是微软为释出开源项目而编写和发布的自由开源软件许可证。如果你用 .NET 开发&#xff0c;你会经常碰见 Ms-PL。在微软的自由开源项目托管地 Codeplex&#xff08;已寿终正寝&#xff0c;微软已战略转…

网络编程-HTTPS协议的实现原理

HTTP传输协议缺点 之前几篇文章中详细讲解了TCP/IP协议栈中的几个协议&#xff0c;其中个就有对HTTP做了一个比较详细的讲解。HTTP是基于TCP进行传输的&#xff0c;其中传输的内容都是明文报文数据&#xff0c;如果我是一个黑客&#xff0c;我会想办法获取这个HTTP消息体&…

从案例角度解析建模平台动态规则引擎

源宝导读&#xff1a;明源云ERP建模平台提供了强大的页面联动规则引擎&#xff0c;原来需要编写代码完成的联动控制逻辑&#xff0c;现在只需要点点鼠标&#xff0c;通过配置完成。本文从实际案例的角度出发&#xff0c;介绍原始的代码逻辑如何转化为引擎规则的过程。一、背景明…

想基于K8s按需扩展应用程序,可从这几方面入手

马修赫瑟&#xff08;Matthew Heusser&#xff09;在花费了十年时间进行编程&#xff0c;测试和项目管理之后&#xff0c;Matt Heusser于2011年创立了自己的公司Excelon Development。该协会前董事会成员Matt还是软件测试人员&#xff0c;是德国波茨坦最具影响力的敏捷测试专业…

windows环境下ELK平台搭建

背景 日志系统主要包括系统日志&#xff0c;应用程序日志和安全日志。系统运维和开发人员可以通过日志了解服务器的软件&#xff0c;硬件信息&#xff0c;检查配置过程中的错误以及错误发生的原因。通常分析日志可以了解服务器的负荷&#xff0c;性能安全性&#xff0c;从而及时…

EntityFramework Core 3.x上下文构造函数可以注入实例呢?

今天讨论的话题来自一位微信好友遇到问题后请求我的帮助&#xff0c;当然他的意图并不是本文标题&#xff0c;只是我将其根本原因进行了一个概括&#xff0c;接下来我们一起来探索标题的问号最终的答案是怎样的呢&#xff1f;老规矩&#xff0c;首先我们定义如下上下文public c…

SpringCloud常见问题总结(一)

Eureka常见问题 Eureka注册服务慢 默认情况&#xff0c;服务注册到Eureka Server 的过程比较慢。在开发或者测试时候&#xff0c;如果能够加速注册的过程&#xff0c;从而提升工作效率。Spring Cloud官方文档详细描述了该问题的原因并提供了解决方案&#xff1a; //原文 Why…

[剑指offer]面试题21:包含min函数的栈

面试题21&#xff1a;包含min函数的栈 题目&#xff1a;定义栈的数据结构&#xff0c;请在该类型中实现一个能够得到栈的最小元素的min函数。在该栈中&#xff0c;调用min、push及pop的时间复杂度都是O&#xff08;1&#xff09;。 栈内压入3、4、2、1之后接连两次弹出栈顶数字…

IO 模型知多少

1. 引言同步异步I/O&#xff0c;阻塞非阻塞I/O是程序员老生常谈的话题了&#xff0c;也是自己一直以来懵懵懂懂的一个话题。比如&#xff1a;何为同步异步&#xff1f;何为阻塞与非阻塞&#xff1f;二者的区别在哪里&#xff1f;阻塞在何处&#xff1f;为什么会有多种IO模型&am…

[剑指offer]面试题22:栈的压入、弹出序列

面试题22&#xff1a;栈的压入、弹出序列 题目&#xff1a;输入两个整数序列&#xff0c;第一个序列表示栈的压入顺序&#xff0c;请判断第二个序列是否为该栈的弹出顺序。假设压入栈的所有数字均不相等。例如序列1、2、3、4、5 是某栈的压栈序列&#xff0c;序列4、5、3、2、1…

Ubuntu上安装使用Docker

Docker简介 Docker是一个开源的容器引擎&#xff0c;他有助于更快的交付应用。Docker可以将应用程序和基础设施层做隔离&#xff0c;并且能将基础设施当做程序一样进行管理。使用Docker&#xff0c;可以更快的打包&#xff0c;测试以及部署应用程序&#xff0c;并且可以缩短从…

让 .NET 轻松构建中间件模式代码

让 .NET 轻松构建中间件模式代码Intro在 asp.net core 中中间件的设计令人叹为观止&#xff0c;如此高大上的设计何不集成到自己的代码里呢。于是就有了封装了一个简单通用的中间件模板的想法&#xff0c;以后有需要的时候就可以拿来即用。接口定义这里按执行的委托是同步还是异…