说到定位和地图,很多人都知道谷歌地图,它在全球范围内有着巨大的影响,在全球大部分地区提供了免费服务,可惜我们中国不在其中。
谷歌地图服务为什么在中国使用不了呢?其实,从技术上来看,并没有什么障碍,谷歌也曾经在中国运行过,主要还是法律和监管等原因,目前在国内是用不了的。
这就给国内的地图和位置服务厂商带来了新的机会,其中高德地图、百度地图、腾讯地图是国内主要的地图厂商,被众多用户采用。但可能很多人不知道的是,还有许多的中小型企业,也能提供地图与定位服务,例如物联网智能硬件终端全域定位服务的维智科技,就提供了一套完整的全域定位SaaS服务平台。
谷歌有提供自己的NLP(Network Location Provider)网络定位,但由于国内网络限制,无法正常使用。目前在国内做底层网络定位的除了高德、百度,就是有着领先技术的维智科技。现在普通安卓用户可能还无法体验NLP的优势,毕竟高德、百度地图已经有强大的功能,完全够各种场景使用。
但对于做专网终端和ROM 的厂商而言,虽然安卓系统自带了定位服务,有GPS、网络定位,但在国内网络定位服务是无法使用的,也就是说在室内或是GPS信号弱等情况无法使用网络定位,只能借助于第三方定位服务商,而高德、百度又不支持私有化场景,这给其他服务商带来了新的契机。
维智深耕位置服务领域,提供NLP定位服务,并支持私有化部署。下面来讲解一下 NLP 定位原理,NLP 一共分为四层,每层都依赖下面一层完成其所需提供的服务。
1. 应用层:是android.location包中的内容,主要通过LocationManager 来调用;
2. 框架层:这一层是系统服务实现,LocationManager 通过Binder 机制来和LocationMagagerService 进行通讯,LocationManagerService 会选择合适的provider 提供位置,基中LocationProviderProxy 的就是NLP,可以理解为LocationProviderProxy和GeocoderProxy都是一个空壳,如果没有第三方实现他们,那么将不提供服务;
3. 公共库层:通过JNI来调用本层libgps.so中的C++ 代码;
4. Linux 内核层:C++代码再通过去与GPS 硬件模块通信获取位置;
通过上图可知GeocoderProxy 与LocationProviderProxy 来调用第三方库来实现网络定位服务以及逆地理解析服务,通过与厂商合作完成包的集成,即可实现定位。常用的LocationProdvider 主要有三种:
1. GPS: 通过手机里面的GPS 芯片,来实现定位;
2. Network: 通过网络来获取位置信息,主要利用手机的基站,和 WIFI 来获取位置;
3. Passive: 是种被动定位方式,它不能自己获取定位方式,是利用被系统保存的其他程序所更新的定位信息。
在应用中触发定位服务,使用方式如下:
通过集成维智NLP,WzLocation.apk,可通过官网注册账号申请,并下载集成开发文档,集成步骤就不详细讲解
1. 申请权限
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<!--后台位置信息 Android 10 (API level 29) 或更高版本 -->
<uses-permission android:name="android.permission.ACCESS_BACKGROUND_LOCATION" />
2. 启动定位
private val gpsLocationManager: LocationManager by lazy { // 初始化对象
getSystemService(Context.LOCATION_SERVICE) as LocationManager
}
private fun getCriteria(): Criteria { // 配置参数
val criteria = Criteria()
criteria.accuracy = Criteria.ACCURACY_FINE
criteria.isSpeedRequired = false
criteria.isCostAllowed = false
criteria.isBearingRequired = false
criteria.isAltitudeRequired = true
criteria.powerRequirement = Criteria.POWER_HIGH
return criteria
}
private fun startLocation() { // 启动定位
info("SurveyLocationManage startLocation ")
gpsLocationManager.let {
if (it.isProviderEnabled(LocationManager.NETWORK_PROVIDER)) {
it.requestLocationUpdates(
LocationManager.NETWORK_PROVIDER,
LOCATION_FIX_INTERVAL, LOCATION_FIX_DISTANCE,
this
)
val bestProvider = it.getBestProvider(getCriteria(), true)
val location = bestProvider?.let { it1 ->
it.getLastKnownLocation(it1)
}
info("startLocation bestProvider $bestProvider")
}
if (it.isProviderEnabled(LocationManager.GPS_PROVIDER)) {
it.requestLocationUpdates(
LocationManager.GPS_PROVIDER,
LOCATION_FIX_INTERVAL, LOCATION_FIX_DISTANCE,
this
)
val bestProvider = it.getBestProvider(getCriteria(), true)
val location = bestProvider?.let { it1 -> it.getLastKnownLocation(it1) }
info("startLocation bestProvider $bestProvider")
}
}
}
3. 回调
override fun onLocationChanged(location: Location) {
var source: String = ""
info("onLocationChanged location $location provider: ${location.provider} " )
if (location.provider == LocationManager.NETWORK_PROVIDER) {
point.text =
"network: longitude : ${location.longitude} latitude : ${location.latitude} accuracy : ${location.accuracy}"
source = "wifi"
dataList.add(location)
adapter.notifyDataSetChanged()
if (location.longitude != 0.0) {
async {
val geocoder = Geocoder(this@LocationActivity)
val addressList =
geocoder.getFromLocation(location.latitude, location.longitude, 5)
//info("onLocationChanged addressList $addressList")
uiThread {
geocode.text = addressList[0].featureName
}
}
}
}
}
4. 通过运行APP 就能显示目标结果
备注:维智NLP 集成接口下载地址:维智 NLP 集成接口文档,通过注册账号,申请AccessKey