NLP网络定位探讨

1,040 阅读3分钟

维智NLP 集成接口下载地址:维智接口集成文档,通过注册账号,申请AccessKey

说到定位和地图,很多人都知道谷歌地图,它在全球范围内有着巨大的影响,在全球大部分地区提供了免费服务,可惜我们中国不在其中。

谷歌地图服务为什么在中国使用不了呢?其实,从技术上来看,并没有什么障碍,谷歌也曾经在中国运行过,主要还是法律和监管等原因,目前在国内是用不了的。

这就给国内的地图和位置服务厂商带来了新的机会,其中高德地图、百度地图、腾讯地图是国内主要的地图厂商,被众多用户采用。但可能很多人不知道的是,还有许多的中小型企业,也能提供地图与定位服务,例如物联网智能硬件终端全域定位服务的维智科技,就提供了一套完整的全域定位SaaS服务平台。

谷歌有提供自己的NLP(Network Location Provider)网络定位,但由于国内网络限制,无法正常使用。目前在国内做底层网络定位的除了高德、百度,就是有着领先技术的维智科技。现在普通安卓用户可能还无法体验NLP的优势,毕竟高德、百度地图已经有强大的功能,完全够各种场景使用。

但对于做专网终端和ROM 的厂商而言,虽然安卓系统自带了定位服务,有GPS、网络定位,但在国内网络定位服务是无法使用的,也就是说在室内或是GPS信号弱等情况无法使用网络定位,只能借助于第三方定位服务商,而高德、百度又不支持私有化场景,这给其他服务商带来了新的契机。

维智深耕位置服务领域,提供NLP定位服务,并支持私有化部署。下面来讲解一下 NLP 定位原理,NLP 一共分为四层,每层都依赖下面一层完成其所需提供的服务。

image.png

  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: 是种被动定位方式,它不能自己获取定位方式,是利用被系统保存的其他程序所更新的定位信息。

在应用中触发定位服务,使用方式如下:

  1. 通过集成维智NLP,WzLocation.apk,可通过官网注册账号申请,并下载集成开发文档,集成步骤就不详细讲解

  2. 申请权限

    <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" />
    
  3. 启动定位

    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")
        }
    }
    
  4. 回调

    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
                    }
                }
            }
        }
    }
    
  5. 通过运行APP 就能显示目标结果

image.png