维智NLP 集成接口下载地址:维智接口集成文档,通过注册账号,申请AccessKey
说到定位和地图,很多人都知道谷歌地图,它在全球范围内有着巨大的影响,在全球大部分地区提供了免费服务,可惜我们中国不在其中。
谷歌地图服务为什么在中国使用不了呢?其实,从技术上来看,并没有什么障碍,谷歌也曾经在中国运行过,主要还是法律和监管等原因,目前在国内是用不了的。
这就给国内的地图和位置服务厂商带来了新的机会,其中高德地图、百度地图、腾讯地图是国内主要的地图厂商,被众多用户采用。但可能很多人不知道的是,还有许多的中小型企业,也能提供地图与定位服务,例如物联网智能硬件终端全域定位服务的维智科技,就提供了一套完整的全域定位SaaS服务平台。
谷歌有提供自己的NLP(Network Location Provider)网络定位,但由于国内网络限制,无法正常使用。目前在国内做底层网络定位的除了高德、百度,就是有着领先技术的维智科技。现在普通安卓用户可能还无法体验NLP的优势,毕竟高德、百度地图已经有强大的功能,完全够各种场景使用。
但对于做专网终端和ROM 的厂商而言,虽然安卓系统自带了定位服务,有GPS、网络定位,但在国内网络定位服务是无法使用的,也就是说在室内或是GPS信号弱等情况无法使用网络定位,只能借助于第三方定位服务商,而高德、百度又不支持私有化场景,这给其他服务商带来了新的契机。
维智深耕位置服务领域,提供NLP定位服务,并支持私有化部署。下面来讲解一下 NLP 定位原理,NLP 一共分为四层,每层都依赖下面一层完成其所需提供的服务。
-
应用层:是android.location包中的内容,主要通过LocationManager 来调用;
-
框架层:这一层是系统服务实现,LocationManager 通过Binder 机制来和LocationMagagerService 进行通讯,LocationManagerService 会选择合适的provider 提供位置,基中LocationProviderProxy 的就是NLP,可以理解为LocationProviderProxy和GeocoderProxy都是一个空壳,如果没有第三方实现他们,那么将不提供服务;
-
公共库层:通过JNI来调用本层libgps.so中的C++ 代码;
-
Linux 内核层:C++代码再通过去与GPS 硬件模块通信获取位置;
通过上图可知GeocoderProxy 与LocationProviderProxy 来调用第三方库来实现网络定位服务以及逆地理解析服务,通过与厂商合作完成包的集成,即可实现定位。
常用的LocationProdvider 主要有三种:
-
GPS: 通过手机里面的GPS 芯片,来实现定位;
-
Network: 通过网络来获取位置信息,主要利用手机的基站,和 WIFI 来获取位置;
-
Passive: 是种被动定位方式,它不能自己获取定位方式,是利用被系统保存的其他程序所更新的定位信息。
在应用中触发定位服务,使用方式如下:
-
通过集成维智NLP,WzLocation.apk,可通过官网注册账号申请,并下载集成开发文档,集成步骤就不详细讲解
-
申请权限
<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" /> -
启动定位
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") } } -
回调
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 } } } } } -
通过运行APP 就能显示目标结果