目录

前言:

1、GPS定位

2、NETWORK定位

3、AGPS定位

4、基站定位

5、WIFI定位

6、混合定位


目前,移动端大致通过三种方式来进行设备定位:GPS、基站、wifi。本文就详细的讲解一下这几种定位方式和实现方法。

前言:

android中我们一般使用LocationManager来获取位置信息,这里面有四中provider:

public static final String NETWORK_PROVIDER = "network";
public static final String GPS_PROVIDER = "gps";
public static final String PASSIVE_PROVIDER = "passive";
public static final String FUSED_PROVIDER = "fused";

区别如下:

(1)GPS_PROVIDER:通过 GPS 来获取地理位置的经纬度信息; 优点:获取地理位置信息精确度高; 缺点:只能在户外使用,获取经纬度信息耗时,耗电;

(2)NETWORK_PROVIDER:通过移动网络的基站或者 Wi-Fi 来获取地理位置; 优点:只要有网络,就可以快速定位,室内室外都可; 缺点:精确度不高;

(3)PASSIVE_PROVIDER:被动接收更新地理位置信息,而不用自己请求地理位置信息。

PASSIVE_PROVIDER 返回的位置是通过其他 providers 产生的,可以查询 getProvider() 方法决定位置更新的由来,需要 ACCESS_FINE_LOCATION 权限,但是如果未启用 GPS,则此 provider 可能只返回粗略位置匹配;

(4)FUSED_PROVIDER:这个本来已经被废弃了,但是目前在Android12(即android api 31)上又重新使用了起来,但是它依赖GMS,所以国内暂时无法使用。

我们通常使用gps和network这两种方式。

但是我们还可以通过其它方式获取位置信息,这篇文章就详细的讲解一下在android中几种获取定位的方式。

1、GPS定位

这个用的最普遍,可以获取上次定位,也可以监听变化,代码如下:

需要权限

<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>

GPS定位

var locManager = getSystemService(Context.LOCATION_SERVICE) as LocationManager
var loc = locManager.getLastKnownLocation(LocationManager.GPS_PROVIDER)
if(loc != null){Log.e("gpslocation", loc.toString())toast(loc.toString())
}
locManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0F, object : LocationListener{override fun onStatusChanged(provider: String?, status: Int, extras: Bundle?) {}override fun onProviderEnabled(provider: String?) {}override fun onProviderDisabled(provider: String?) {}override fun onLocationChanged(location: Location?) {Log.e("gpslocation", location.toString())toast(location.toString())}
})

2、NETWORK定位

与gps定位代码基本一致,只不过将provider改成LocationManager.NETWORK_PROVIDER

3、AGPS定位

实际上是将上面两种定位结合起来,具体原理如下:

AGPS手机首先将本身的基站地址通过网络传输到位置服务器;

位置服务器根据该手机的大概位置传输与该位置相关的GPS辅助信息(包含GPS的星历和方位俯仰角等)到手机;

该手机的AGPS模块根据辅助信息(以提升GPS信号的第一锁定时间TTFF能力)接收GPS原始信号;

手机在接收到GPS原始信号后解调信号,计算手机到卫星的伪距(伪距为受各种GPS误差影响的距离),并将有关信息通过网络传输到位置服务器;

位置服务器根据传来的GPS伪距信息和来自其他定位设备(如差分GPS基准站等)的辅助信息完成对GPS信息的处理,并估算该手机的位置;

位置服务器将该手机的位置通过网络传输到定位网关或应用平台。

我的理解就是通过网络位置和位置服务器判断出最佳的卫星,减少了获取卫星信号的时间。因为网络位置获取很快,所以可以减少整体的定位时间。

AGPS并不是一种定位方式,只是一种优化方案,代码与GPS一样,只不过在设置中将定位模式设成AGPS。


上面是android自带的定位方式,我们还可以获取一些原始信息(比如基站信息、wifi信息),通过公开的接口来获取位置信息。下面几种方式就是使用原始信息通过API来获取位置信息。

4、基站定位

通过TelephonyManager我们可以拿到基站信息,再通过相关的api接口就能得到经纬度,但是基站定位精度很差。

基站信息包含如下:

MCC,Mobile Country Code,移动国家代码(中国的为460);
MNC,Mobile Network Code,移动网络号码(中国移动为00,中国联通为01);
LAC,Location Area Code,位置区域码;
CID,Cell Identity,基站编号,是个16位的数据(范围是0到65535)。

拿到这个信息后,我们可以通过一些公开的api服务拿到经纬度,如下:

http://www.google.com/loc/json   google的,post请求,好像停用了

接口说明文档 | LBS数据仓库  目前可用,是免费的

实例:

http://api.cellocation.com:82/cell/?mcc=460&mnc=1&lac=4301&ci=20986&output=json

返回:

{"errcode":0, "lat":"40.00598145", "lon":"116.48539734", "radius":"937", "address":"北京市朝阳区来广营地区东湖渠;溪阳东路与屏翠东路路口东70米"}

接口参数:

名称 类型 必填 说明
mcc int mcc国家代码:中国代码 460
mnc int mnc网络类型:0移动,1联通(电信对应sid),十进制
lac int lac(电信对应nid),十进制
ci int cellid(电信对应bid),十进制
coord string 坐标类型(wgs84/gcj02/bd09),默认wgs84
output string 返回格式(csv/json/xml),默认csv

另外还有很多提供这种接口和数据的平台,自己搜索即可

代码如下:

需要权限

<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>

基站定位,这里只实现了GSM的,CDMA的有些许不同

val telManager = getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManager
if(telManager.cellLocation is GsmCellLocation) {val cellLoc = telManager.cellLocation as GsmCellLocationif (cellLoc != null) {val operator = telManager.networkOperatorval mcc = operator.substring(0, 3).toInt()val mnc = operator.substring(3).toInt()val cid = cellLoc.cidval lac = cellLoc.lac...doAsync {val result = URL(sb.toString()).readText()Log.e("tellocation", result)}}
}

5、WIFI定位

wifi定位是通过WifiManager拿到wifi的信息,主要是wifi的BSSID(即mac地址)。然后通过一些api查询经纬度,比如接口说明文档 | LBS数据仓库

实例:

http://api.cellocation.com:82/wifi/?mac=00:87:36:05:5d:ea&output=json

返回:

{"errcode":0, "lat":"39.950008", "lon":"116.230049", "radius":"222", "address":"北京市海淀区四季青镇益园文创基地c区9号楼;南平庄中路与西平庄路路口西北561米"}

接口参数:

名称 类型 必填 说明
mac string WIFI热点的MAC地址(BSSID)
coord string 坐标类型(wgs84/gcj02/bd09),默认wgs84
output string 返回格式(csv/json/xml),默认csv

代码如下:

需要权限

<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>

wifi定位

val wifiManager = applicationContext.getSystemService(Context.WIFI_SERVICE) as WifiManager
if(wifiManager.isWifiEnabled) {var mac = wifiManager.connectionInfo.bssidif(TextUtils.isEmpty(mac)) {/*** 当未链接wifi时,可以使用扫描到的wifi列表中找一个信号强度最好的。这里没进行比较,直接使用第一个了* ScanReuslt有三个字段比较重要:SSID是wifi名称,BSSID是wifi的mac,level则是信号强度(负数)* 注意结果中同一个SSID可能会有多个,如果需要链接wifi可以通过信号强度过滤出最好的来链接*/val scanlist = wifiManager.scanResultsfor(info in scanlist){Log.e("wifiinfo", info.toString())}if(scanlist.size > 0) {mac = wifiManager.scanResults[0].BSSID}}if(!TextUtils.isEmpty(mac)) {...doAsync {val result = URL(sb.toString()).readText()Log.e("wifilocation", result)}}
}

6、混合定位

混合定位就是获取附近的wifi列表信息(包括信号强度)和附近的基站列表信息(包括信号强度),通过一些api获取经纬度。

这种方式相对于单一的基站和wifi定位要更精确一些。

获取附近的wifi列表在WIFI定位已经提到过了,通过WifiManager的getScanResults函数获取扫描到的wifi列表,其中level就是信号强度,可能需要做一下去重。

获取附近的基站列表则有些问题。

官方提供了一个方式,通过TelephonyManager的getNeighboringCellInfo函数获得,其中mRssi就是信号强度。

需要权限

<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>

获取附近基站信息

val telManager = getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManager
val neighborCells = telManager.neighboringCellInfo
for(cell in neighborCells){//这里将rssi转化为dBmval level = -131 + 2 * cell.rssiLog.e("neighboringCellInfo", "cid:${cell.cid} lac:${cell.lac} rssi:$level")
}

但是实际使用时发现cid和lac都是-1。

检查NeighboringCellInfo的构造方法,如下:

 public NeighboringCellInfo(int rssi, String location, int radioType) {// set default valuemRssi = rssi;mNetworkType = NETWORK_TYPE_UNKNOWN;mPsc = UNKNOWN_CID;mLac = UNKNOWN_CID;mCid = UNKNOWN_CID;// pad location string with leading "0"int l = location.length();if (l > 8) return;if (l < 8) {for (int i = 0; i < (8-l); i++) {location = "0" + location;}}// TODO - handle LTE and eHRPD (or find they can't be supported)try {// set LAC/CID or PSC based on radioTypeswitch (radioType) {case NETWORK_TYPE_GPRS:case NETWORK_TYPE_EDGE:mNetworkType = radioType;// check if 0xFFFFFFFF for UNKNOWN_CIDif (!location.equalsIgnoreCase("FFFFFFFF")) {mCid = Integer.parseInt(location.substring(4), 16);mLac = Integer.parseInt(location.substring(0, 4), 16);}break;case NETWORK_TYPE_UMTS:case NETWORK_TYPE_HSDPA:case NETWORK_TYPE_HSUPA:case NETWORK_TYPE_HSPA:mNetworkType = radioType;mPsc = Integer.parseInt(location, 16);break;}} catch (NumberFormatException e) {// parsing location errormPsc = UNKNOWN_CID;mLac = UNKNOWN_CID;mCid = UNKNOWN_CID;mNetworkType = NETWORK_TYPE_UNKNOWN;}}

可以看到只对GPRS和EDGE网络进行了处理,而3G、4G网络都是UNKNOWN_CID,即-1。说明这种方法不支持,已经过时。

官方还有另外一个方式,通过TelephonyManager的getAllCellInfo函数获得。这个函数要求minsdkverison必须在17及以上

需要权限

<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>

获取基站信息,这里只处理了LTE网络的

val telManager = getSystemService(Context.TELEPHONY_SERVICE) as TelephonyManager
val cells = telManager.allCellInfo
for(cell in cells){if(cell is CellInfoLte) {Log.e("celllist", "cid:${cell.cellIdentity.ci} lac:${cell.cellIdentity.tac} dbm:${cell.cellSignalStrength.dbm} isRegistered:${cell.isRegistered}")}
}

得到的信息如下:

E/celllist: cid:3912981 lac:4154 dbm:-87 isRegistered:true

E/celllist: cid:2147483647 lac:2147483647 dbm:-101 isRegistered:false

E/celllist: cid:2147483647 lac:2147483647 dbm:-102 isRegistered:false

E/celllist: cid:2147483647 lac:2147483647 dbm:-108 isRegistered:false

E/celllist: cid:2147483647 lac:2147483647 dbm:-101 isRegistered:false

其中第一个是我们正在使用的基站,可以看到正常的返回了信息,而其余的则返回默认的信息(Integer.MAX_VALUE)

说明这个方法也只能拿到当前使用的基站信息。而且据网上的说法,当使用2G网络,getAllCellInfo得到的是NULL。

这样目前没有更好的方式获取多个基站信息了。

当我们拿到附近的基站信息和wifi信息,可以通过http://www.cellocation.com/interfac/ 提供的混合定位接口查询位置信息,与上面类似,这里不细说了。

总结

一般情况下,我们使用系统提供的LocationManager即可获取位置信息,方便简单。如果我们有自己的基站或wifi信息库,也可以获取相关源信息通过接口来实现个性化服务。

源码

关注公众号:BennuCTech,发送“多定位”获取源码

android中几种定位方式详解相关推荐

  1. PCBA加工中常见的两种焊接方式详解

    PCBA加工中常见的两种焊接方式详解 PCBA加工,两种常见的焊接方式就是回流焊和波峰焊,与手动焊接技术相比,自动焊接技术具有减少人为因素的影响.提高效率.降低成本.提高质量等优势,在PCBA加工中, ...

  2. python定位相邻节点_Python selenium 父子、兄弟、相邻节点定位方式详解

    今天跟大家分享下selenium中根据父子.兄弟.相邻节点定位的方法,很多人在实际应用中会遇到想定位的节点无法直接定位,需要通过附近节点来相对定位的问题,但从父节点定位子节点容易,从子节点定位父节点. ...

  3. @Resource,@Autowired,@Inject3种注入方式详解

    转载自 @Resource,@Autowired,@Inject3种注入方式详解 概况 @Resource,@Autowired,@Inject 这3种都是用来注入bean的,它们属于不同的程序中. ...

  4. python编程midi键盘按键_Python中捕获键盘的方式详解

    python中捕获键盘操作一共有两种方法 第一种方法: 使用pygame中event方法 使用方式如下:使用键盘右键为例 if event.type = pygame.KEYDOWN and even ...

  5. python获取键盘按键_Python中捕获键盘的方式详解

    python中捕获键盘操作一共有两种方法 第一种方法: 使用pygame中event方法 使用方式如下:使用键盘右键为例 if event.type = pygame.KEYDOWN and even ...

  6. Hibernate中的QBC查询方式详解

    Hibernate中的QBC查询方式详解 QBC:Query By Criteria,条件查询. 是一种更加面向对象化的查询的方式. 1.QBC简单查询 测试代码: package com.pipi. ...

  7. C++的三种继承方式详解

    文章目录 @[toc] C++的三种继承方式详解以及区别 前言 一.public继承 二.protected继承 三.private继承 四.三者区别 五.总结 后话 C++的三种继承方式详解以及区别 ...

  8. Android中measure过程、WRAP_CONTENT详解以及xml布局文件解析流程浅析(下)

       本文原创, 转载请注明出处:http://blog.csdn.net/qinjuning 上篇文章<<Android中measure过程.WRAP_CONTENT详解以及xml布局文 ...

  9. android 中的悬浮按钮,Android 中FloatingActionButton(悬浮按钮)实例详解

    android 中floatingactionbutton(悬浮按钮)实例详解 一.介绍 这个类是继承自imageview的,所以对于这个控件我们可以使用imageview的所有属性 二.使用准备, ...

最新文章

  1. java泛型的实现和原理_java 泛型实现原理
  2. 熬10天夜,肝出了这个PDF版“软件安装手册”(附下载)
  3. 『中级篇』k8s的NodePort类型Service以及Label的简单实用(68)
  4. mysql主从复制及失败切换
  5. Linux显示某文件中有关某字符串的信息
  6. Linux - which xxx - 查找执行的命令所在的路径
  7. php 多态有什么用,php面向对象多态的介绍与优势
  8. jpa怎么传参到in中_Java中如何处理开关状态的属性字段?
  9. python 模拟登录博客园并且自动发布一篇文章
  10. modbus测试plc软件,MODBUS RTU设备测试调试工具官方版
  11. python实现t分布
  12. 关于Windows 10 企业版 LTSC重装系统后优化项目
  13. RHEL7 CentOS7 检查查看精简指令
  14. Win10插入U盘无反应,但是U盘是正常的解决方法
  15. 上海市新能源汽车分时租赁规划策略研究
  16. 【学习笔记】图像纹理特征
  17. Python实现《都挺好》社交网络分析
  18. (P61)io流类库:字符串流
  19. leetcode-滑动窗口(双指针-代码有套路-附总结代码)总结-满足条件-破坏条件(76,438,3,209有模板(别人总结的很好,学习一波))
  20. 【微服务】165:导入数据到索引库

热门文章

  1. Google昨天发布的新产品——Google Music
  2. 多操作系统安装实践小结
  3. git移除某文件夹的版本控制
  4. BZOJ 3224: Tyvj 1728 普通平衡树【Treap】
  5. Python-基础知识-控制流程和文件操作
  6. Oracle数据库之事务
  7. Mac下的Mysql无法登陆的问题
  8. 提示用户输入一个正整数n,如果n=5,就输出下列图形,其他n值以此类推
  9. C语言中不检查数组下标是否越界。
  10. 64位以内Rabin-Miller 强伪素数测试和Pollard rho 因数分解解析