博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
HTML5 地理位置定位(HTML5 Geolocation)原理及应用 (调用GPS)
阅读量:6497 次
发布时间:2019-06-24

本文共 7661 字,大约阅读时间需要 25 分钟。

hot3.png

博客分类:

 

 

地理位置(Geolocation)是  的重要特性之一,提供了确定用户位置的功能,借助这个特性能够开发基于位置信息的应用。今天这篇文章向大家介绍一下  地理位置定位的基本原理及各个浏览器的数据精度情况。

在 访问位置信息前,浏览器都会询问用户是否共享其位置信息,以 Chrome 浏览器为例,如果您允许 Chrome 浏览器与网站共享您的位置,Chrome 浏览器会向 Google 位置服务发送本地网络信息,估计您所在的位置。然后,浏览器会与请求使用您位置的网站共享您的位置。

HTML5 Geolocation API 使用非常简单,基本调用方式如下:

Js代码  收藏代码

  1. if (navigator.geolocation) {  

  2.     navigator.geolocation.getCurrentPosition(locationSuccess, locationError,{  

  3.         // 指示浏览器获取高精度的位置,默认为false  

  4.         enableHighAcuracy: true,  

  5.         // 指定获取地理位置的超时时间,默认不限时,单位为毫秒  

  6.         timeout: 5000,  

  7.         // 最长有效期,在重复获取地理位置时,此参数指定多久再次获取位置。  

  8.         maximumAge: 3000  

  9.     });  

  10. }else{  

  11.     alert("Your browser does not support Geolocation!");  

  12. }  

 

locationError为获取位置信息失败的回调函数,可以根据错误类型提示信息:

Js代码  收藏代码

  1. locationError: function(error){  

  2.     switch(error.code) {  

  3.         case error.TIMEOUT:  

  4.             showError("A timeout occured! Please try again!");  

  5.             break;  

  6.         case error.POSITION_UNAVAILABLE:  

  7.             showError('We can\'t detect your location. Sorry!');  

  8.             break;  

  9.         case error.PERMISSION_DENIED:  

  10.             showError('Please allow geolocation access for this to work.');  

  11.             break;  

  12.         case error.UNKNOWN_ERROR:  

  13.             showError('An unknown error occured!');  

  14.             break;  

  15.     }  

  16. }  

 

locationSuccess为获取位置信息成功的回调函数,返回的数据中包含经纬度等信息,结合Google Map API 即可在地图中显示当前用户的位置信息,如下:

Js代码  收藏代码

  1. locationSuccess: function(position){  

  2.     var coords = position.coords;       

  3.     var latlng = new google.maps.LatLng(  

  4.         // 维度  

  5.         coords.latitude,  

  6.         // 精度  

  7.         coords.longitude  

  8.     );     

  9.     var myOptions = {     

  10.         // 地图放大倍数     

  11.         zoom: 12,     

  12.         // 地图中心设为指定坐标点     

  13.         center: latlng,     

  14.         // 地图类型     

  15.         mapTypeId: google.maps.MapTypeId.ROADMAP     

  16.     };     

  17.     // 创建地图并输出到页面     

  18.     var myMap = new google.maps.Map(     

  19.         document.getElementById("map"),myOptions     

  20.     );     

  21.     // 创建标记     

  22.     var marker = new google.maps.Marker({     

  23.         // 标注指定的经纬度坐标点     

  24.         position: latlng,     

  25.         // 指定用于标注的地图     

  26.         map: myMap  

  27.     });  

  28.     //创建标注窗口     

  29.     var infowindow = new google.maps.InfoWindow({     

  30.         content:"您在这里<br/>纬度:"+     

  31.             coords.latitude+     

  32.             "<br/>经度:"+coords.longitude     

  33.     });     

  34.     //打开标注窗口     

  35.     infowindow.open(myMap,marker);    

  36. }  

 

 

经过测试,Chrome/Firefox/Safari/Opera四个浏览器获取到的位置信息都是一摸一样的,估计都是用的同一个位置服务,数据如下:

 

 

位置服务用于估计您所在位置的本地网络信息包括:有关可见 WiFi 接入点的信息(包括信号强度)、有关您本地路由器的信息、您计算机的 IP 地址。位置服务的准确度和覆盖范围因位置不同而异。

总的来说,在PC的浏览器中  的地理位置功能获取的位置精度不够高,如果借助这个 HTML5 特性做一个城市天气预报是绰绰有余,但如果是做一个地图应用,那误差还是太大了。不过,如果是移动设备上的  应用,可以通过设置 enableHighAcuracy 参数为 true,调用设备的 GPS 定位来获取高精度的地理位置信息。

原文:

 

请求用户许可

Js代码  收藏代码

  1. navigator.geolocation.getCurrentPosition(on_success, on_error, options);  

getCurrentPosition包含三个参数,前两个为函数名,第三个为一个对象。其中只有第一个是必须的。当你执行上面的 JavaScript 语句后,浏览器通常会弹出一个提示,询问用户是否允许网站跟踪位置信息;同时getCurrentPosition函数会立即返回。如果用户选择了允许,则会执行上述on_success函数,这时你才真正得到位置信息(这就是这件事情为什么要分两步的原因——用户需要一定时间才能对请求作出反应,同时地理位置信息可能需要一定时间才能生成,而函数需要立即返回)。

 

错误处理

其中error.code为一个枚举类型,可能的取值如下:

  • PERMISSION_DENIED:用户拒绝

  • POSITION_UNAVAILABLE:地理位置获取失败(可能是用户没网或卫星搜不到等原因)

  • TIMEOUT:地理位置获取超时

error.message则为一个可以帮助开发者调试的错误信息(此信息一般不适合直接显示在网页中给用户查看)。

 

可选项

事实上,上述getCurrentPosition函数还支持第三个可选的参数,是一个 Option Object,一共有三个选项可以设定:

Js代码  收藏代码

  1. var options = {  

  2.     enableHighAccuracy: false,  

  3.     timeout: 5000,  

  4.     maximumAge: 60000  

  5. }  

 

其中timeout是设定地理位置获取的超时时间(单位为毫秒,用户选择允许的时间不计算在内);而maximumAge表示允许设备从缓存中读取位置,缓存的过期时间,单位是毫秒,设为0来禁用缓存读取。如果返回的是缓存中的时间,会在timestamp中反映出来。

 

兼容性列表

支持 Geolocation API 的浏览器/终端/操作系统:

  • Firefox 3.5+

  • Google Chrome 5.0+

  • Safari 5.0+

  • Opera 10.60+

  • Internet Explorer 9.0+

  • Android 2.0+

  • iOS 3.0+

  • Opera Mobile 10.1+

  • Blackberry OS 6

 

HTML5 Geolocation API的使用方法及实现原理

上周项目不忙,抽时间研究了一下HTML5的geolocation。

在HTML5中,geolocation作为navigator的一个属性出现,它本身是一个对象,拥有三个方法:

- getCurrentPosition

- watchPosition

- clearWatch

具体用法如下:

Js代码  收藏代码

  1. //判断浏览器是否支持geolocation  

  2. if(navigator.geolocation){  

  3.      // getCurrentPosition支持三个参数  

  4.      // getSuccess是执行成功的回调函数  

  5.      // getError是失败的回调函数  

  6.      // getOptions是一个对象,用于设置getCurrentPosition的参数  

  7.      // 后两个不是必要参数  

  8.      var getOptions = {  

  9.           //是否使用高精度设备,如GPS。默认是true  

  10.           enableHighAccuracy:true,  

  11.           //超时时间,单位毫秒,默认为0  

  12.           timeout:5000,  

  13.           //使用设置时间内的缓存数据,单位毫秒  

  14.           //默认为0,即始终请求新数据  

  15.           //如设为Infinity,则始终使用缓存数据  

  16.           maximumAge:0  

  17.      };  

  18.    

  19.      //成功回调  

  20.      function getSuccess(position){  

  21.           // getCurrentPosition执行成功后,会把getSuccess传一个position对象  

  22.           // position有两个属性,coords和timeStamp  

  23.           // timeStamp表示地理数据创建的时间??????  

  24.           // coords是一个对象,包含了地理位置数据  

  25.           console.log(position.timeStamp);     

  26.    

  27.           // 估算的纬度  

  28.           console.log(position.coords.latitude);      

  29.           // 估算的经度  

  30.           console.log(position.coords.longitude);      

  31.           // 估算的高度 (以米为单位的海拔值)  

  32.           console.log(position.coords.altitude);      

  33.           // 所得经度和纬度的估算精度,以米为单位  

  34.           console.log(position.coords.accuracy);      

  35.           // 所得高度的估算精度,以米为单位  

  36.           console.log(position.coords.altitudeAccuracy);      

  37.           // 宿主设备的当前移动方向,以度为单位,相对于正北方向顺时针方向计算  

  38.           console.log(position.coords.heading);  

  39.           // 设备的当前对地速度,以米/秒为单位      

  40.           console.log(position.coords.speed);      

  41.           // 除上述结果外,Firefox还提供了另外一个属性address  

  42.           if(position.address){  

  43.                //通过address,可以获得国家、省份、城市  

  44.                console.log(position.address.country);  

  45.                console.log(position.address.province);  

  46.                console.log(position.address.city);  

  47.           }  

  48.      }  

  49.      //失败回调  

  50.      function getError(error){  

  51.           // 执行失败的回调函数,会接受一个error对象作为参数  

  52.           // error拥有一个code属性和三个常量属性TIMEOUT、PERMISSION_DENIED、POSITION_UNAVAILABLE  

  53.           // 执行失败时,code属性会指向三个常量中的一个,从而指明错误原因  

  54.           switch(error.code){  

  55.                case error.TIMEOUT:  

  56.                     console.log('超时');  

  57.                     break;  

  58.                case error.PERMISSION_DENIED:  

  59.                     console.log('用户拒绝提供地理位置');  

  60.                     break;  

  61.                case error.POSITION_UNAVAILABLE:  

  62.                     console.log('地理位置不可用');  

  63.                     break;  

  64.                default:  

  65.                     break;  

  66.           }  

  67.      }  

  68.   

  69.      navigator.geolocation.getCurrentPosition(getSuccess, getError, getOptions);  

  70.      // watchPosition方法一样可以设置三个参数  

  71.      // 使用方法和getCurrentPosition方法一致,只是执行效果不同。  

  72.      // getCurrentPosition只执行一次  

  73.      // watchPosition只要设备位置发生变化,就会执行  

  74.      var watcher_id = navigator.geolocation.watchPosition(getSuccess, getError, getOptions);  

  75.      //clearwatch用于终止watchPosition方法  

  76.      navigator.geolocation.clearWatch(watcher_id);           

  77. }  

 

 

geolocation的使用方法并不复杂,但是其实现原理比较有意思。

通过观察geolocation的使用方法,可以发现这个api可以在用户允许的情况下调用很多系统设备,比如GPS、WIFI、蓝牙等。

W3C对geolocation的定义是这样的:

The Geolocation API defines a high-level interface to location information associated only with the device hosting the implementation, such as latitude and longitude. The API itself is agnostic of the underlying location information sources. Common sources of location information include Global Positioning System (GPS) and location inferred from network signals such as IP address, RFID, WiFi and Bluetooth MAC addresses, and GSM/CDMA cell IDs, as well as user input. No guarantee is given that the API returns the device's actual location.

这 里提到了,geolocation的位置信息来源包括GPS、IP地址、RFID、WIFI和蓝牙的MAC地址、以及GSM/CDMS的ID等等。规范 中没有规定使用这些设备的先后顺序,在HTML5的实现中,手机等移动设备当然优先使用GPS定位,而笔记本和部分平板,最准的定位是WIFI,至于网线 上网的台式机,一般就只能使用IP来定位了,这个准确度最低。

在这些方法里,GPS定位最好理解,卫星直接给出定 位数据。而WIFI和IP地址定位,都不是浏览器本身能够实现的。这两种方式都必须将IP地址或 WIFI信号收集到的周围路由信息,上传到某个服务器,由服务器的查询计算位置信息,然后返回给浏览器。那么这些查询服务由谁来提供呢?

首 先来看chrome,很明显,肯定是google自己提供的服务。通过chrome自带的抓包方法(chrome://net-internals/) 可以看到,在使用geolocation时,chrome向www.googleapis.com/geolocation/v1/geolocate的 接口发送了请求,由于请求用spdy加密过,所以看不出具体内容,只有一点可以确定,即wifi上网时,是post方式传数据,而使用网线时,使用的是 get。

firefox使用的也是google的服务,但是和chrome用的接口不同,这个是https://maps.googleapis.com/maps/api/browserlocation/json

请 求数据是:browser=firefox&sensor=true&wifi=mac:xx-xx-xx-xx-xx- xx%7Cssid:xxxxxxx%7Css:-43&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-43&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-43&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-44&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-60&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-61&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-62&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-63&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-63&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-67&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-67&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-67&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-67&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-74&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-82&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-84&wifi=mac:xx-xx-xx-xx-xx-xx%7Cssid:xxxxxxx%7Css:-85

这个就很清晰了,是周边wifi设备的ssid、mac地址,以及信号强度。(公司网络,路由名和mac地址隐去。)

opera用的也是google的服务,接口url没弄到,不过从ip地址来看,也是google的ip。

对于不支持geolocation的浏览器,本来可以调用google的gears项目提供的接口来查询地理位置,但是该服务目前2011年已经停止,暂时也没有出现更好的替代方案。

转载于:https://my.oschina.net/u/615191/blog/607626

你可能感兴趣的文章
MVC设计模式
查看>>
在团队项目遇到的问题及解决方法。
查看>>
springcloud demo---config-client
查看>>
Django设置联合唯一约束 -- migrate时报错处理
查看>>
Java LeetCode 1.Two Sum
查看>>
前端面试题:css相关面试题
查看>>
shell命令的高级使用之---选择性copy
查看>>
最长回文子序列-----动态规划
查看>>
Vue国际化实现
查看>>
设计模式:单例模式
查看>>
FLASH位宽为8、16、32时,CPU与外设之间地址线的连接方法
查看>>
双网卡一般情况不能有两个网关 (转)
查看>>
xshell 远程连接Linux
查看>>
Linux计划任务及压缩归档(week2_day1)--技术流ken
查看>>
ccf算法模板
查看>>
微信小程序登录 该死的官方文档TypeError: the JSON object must be str, not 'bytes'
查看>>
VMware 虚拟机克隆 CentOS 6.5 之后,网络配置问题的解决方案
查看>>
Python ( 1 ) ----- 简介
查看>>
[linux基础学习]run level
查看>>
第七周学习总结
查看>>