Tonfay
2014-08-06 14:17:35
mysql经纬度查询并且计算2KM范围内附近用户的sql查询性能优化实例教程
之前很傻很天真地以为无非就是逐个计算距离,然后比较出来就行了,然后当碰到访问用户很多,而且数据库中经纬度信息很多的时候,计算量的迅速增长,能让服务器完全傻逼掉,还是老前辈的经验比我们丰富,给了我很大的启示。
MySQL性能调优 – 使用更为快速的算法进行距离计算
最近遇到了一个问题,通过不断的尝试最终将某句原本占据近1秒的查询优化到了0.01秒,效率提高了100倍.
问题是这样的,有一张存放用户居住地点经纬度信息的MySQL数据表,表结构可以简化 为:id(int),longitude(long),latitude()long. 而业务系统中有一个功能是查找离某个用户最近的其余数个用户,通过代码分析,可以确定原先的做法基本是这样的:
//需要查询的用户的坐标
$lat=20; $lon=20;//执行查询,算出该用户与所有其他用户的距离,取出最近的10个 $sql='select * from users_location order by ACOS(SIN(('.$lat.' * 3.1415) / 180 ) *SIN((latitude * 3.1415) / 180 ) +COS(('.$lat.' * 3.1415) / 180 ) * COS((latitude * 3.1415) / 180 ) *COS(('.$lon.' * 3.1415) / 180 - (longitude * 3.1415) / 180 ) ) * 6380 asc limit 10';
而这条sql执行的速度却非常缓慢,用了近1秒的时间才返回结果,应该是因为order里的子语句用了太多的数学计算公式,导致整体的运算速度下降.
而在实际的使用中,不太可能会发生需要计算该用户与所有其他用户的距离,然后再排序的情况,当用户数量达到一个级别时,就可以在一个较小的范围里进行搜索,而非在所有用户中进行搜索.
所以对于这个例子,我增加了4个where条件,只对于经度和纬度大于或小于该用户1度(111公里)范围内的用户进行距离计算,同时对数据表中的经度和纬度两个列增加了索引来优化where语句执行时的速度.
最终的sql语句如下
$sql='select * from users_location where latitude > '.$lat.'-1 and latitude < '.$lat.'+1 and longitude > '.$lon.'-1 and longitude < '.$lon.'+1 order by ACOS(SIN(('.$lat.' * 3.1415) / 180 ) *SIN((latitude * 3.1415) / 180 ) +COS(('.$lat.' * 3.1415) / 180 ) * COS((latitude * 3.1415) / 180 ) *COS(('.$lon.'* 3.1415) / 180 - (longitude * 3.1415) / 180 ) ) * 6380 asc limit 10';
经过优化的sql大大提高了运行速度,在某些情况下甚至有100倍的提升.这种从业务角度出发,缩小sql查询范围的方法也可以适用在其他地方.
原文地址: http://blog.csdn.net/hustpzb/article/details/7688993
猜你喜欢
- mysql创建多列索引查询百万表数据的性能优化经验分享
- Mysql低效的where 1=1性能测试
- echart通过jsp连接查询mysql的demo
- Spring+Spring MVC+MyBatis整合实现junit查询mysql数据库
- java HashMap四种遍历查询方式耗时性能对比
- java程序中字符串比较常见的四种判断是否为空的性能优化比较
- 最代码网站用户私信列表采用mysql union查询优化为Redis查询的经验和相关代码片段分享
- java读取带有地理位置照片的信息
- Java备份恢复Mysql
- 【LBS】移动互联网基于LBS地理位置应用java开发必备
- mySql启动和关闭.bat文件(需要管理员权限)
- 我写的备份mysql数据库的代码
请下载代码后再发表评论
相关代码
- mySql启动和关闭.bat文件(需要管理员权限)
- Java备份恢复Mysql
- Mysql低效的where 1=1性能测试
- 我写的备份mysql数据库的代码
- mysql数据库动态添加字段设计
- Java在线备份和还原MySQL数据库脚本代码
- 原证精 mysql创建多列索引查询百万表数据的性能优化经验分享
- java直接连接mysql数据库的代码
- windows下利用任务计划备份mysql数据库
- 原证 Spring mvc整合mybatis基于mysql数据库实现用户增删改查及其分页显示的完整入门实例
- 原 apache log4j实现日志插入mysql数据库
- 证精 将java log4j的日志配置输出到mysql中
最近下载
最近浏览
liulian666 LV2
7月19日
3334004690 LV10
3月6日
卢本伟不开挂 LV4
2023年9月3日
矿泉水 LV30
2023年8月11日
lironggang LV38
2023年4月25日
微信网友_5992582549164032 LV6
2022年12月12日
nurmamat001 LV2
2022年11月1日
wangjie49 LV7
2022年10月1日
xyz_dself LV2
2022年9月30日
123456dswhdm LV2
2022年8月22日