写点什么

融合多元定位技术,帮助应用破解精准定位难题

作者:HarmonyOS SDK
  • 2025-09-26
    贵州
  • 本文字数:2434 字

    阅读完需:约 8 分钟

从查阅天气到记录运动轨迹,从打车到路径规划等,移动终端已深入人们日常生活的方方面面,看似简单的操作背后,都依赖于精准的定位技术。对于应用开发者,位置数据的实时性与准确性直接决定用户体验。当前主流定位技术正面临两难选择:GPS 等卫星定位虽能实现<5 米的高精度,却需长时间开启硬件模块,导致设备耗电量激增,且冷启动时受卫星信号搜索、数据计算影响,等待时长普遍达 10-30 秒;而基站/Wi-Fi 定位虽能秒级响应并降低功耗,但精度可能扩大至百米级,难以支撑精准导航等深度场景。


HarmonyOS SDK位置服务(Location Kit)的位置子系统使用多种定位技术提供服务,如 GNSS 定位、基站定位、WLAN/蓝牙定位(基站定位、WLAN/蓝牙定位后续统称"网络定位技术");通过这些定位技术,无论用户设备在室内或是户外,都可以较为准确地确定设备位置。


Location Kit 除了提供基础的定位服务之外,还提供了地理围栏、地理编码、逆地理编码、国家码等功能和接口。


开发步骤

  1. 获取设备的位置信息,需要有位置权限,位置权限申请的方法和步骤见申请位置权限开发指导

  2. 导入 geoLocationManager 模块,所有与基础定位能力相关的功能 API,都是通过该模块提供的。

  3. import { geoLocationManager } from '@kit.LocationKit';

  4. 调用获取位置接口之前需要先判断位置开关是否打开。


查询当前位置开关状态,返回结果为布尔值,true 代表位置开关开启,false 代表位置开关关闭,示例代码如下:


import { geoLocationManager } from '@kit.LocationKit';try {    let locationEnabled = geoLocationManager.isLocationEnabled();} catch (err) {    console.error("errCode:" + err.code + ", message:"  + err.message);}
复制代码


如果位置开关未开启,可以拉起全局开关设置弹框,引导用户打开位置开关,具体可参考拉起全局开关设置弹框。


  1. 单次获取当前设备位置。多用于查看当前位置、签到打卡、服务推荐等场景。


方式一:获取系统缓存的最新位置。


如果系统当前没有缓存位置会返回错误码。


推荐优先使用该接口获取位置,可以减少系统功耗。


如果对位置的新鲜度比较敏感,可以先获取缓存位置,将位置中的时间戳与当前时间对比,若新鲜度不满足预期可以使用方式二获取位置。


import { geoLocationManager } from '@kit.LocationKit';import { BusinessError } from '@kit.BasicServicesKit'try {    let location = geoLocationManager.getLastLocation();} catch (err) {    console.error("errCode:" + JSON.stringify(err));}
复制代码


方式二:获取当前位置。


首先要实例化 SingleLocationRequest 对象,用于告知系统该向应用提供何种类型的位置服务,以及单次定位超时时间。


  1. 设置 LocatingPriority:


如果对位置的返回精度要求较高,建议 LocatingPriority 参数优先选择 PRIORITY_ACCURACY,会将一段时间内精度较好的结果返回给应用。


如果对定位速度要求较高,建议 LocatingPriority 参数选择 PRIORITY_LOCATING_SPEED,会将最先拿到的定位结果返回给应用。


两种定位策略均会同时使用 GNSS 定位和网络定位技术,以便在室内和户外场景下均可以获取到位置结果,对设备的硬件资源消耗较大,功耗也较大。


  1. 设置 locatingTimeoutMs:


因为设备环境、设备所处状态、系统功耗管控策略等的影响,定位返回的时延会有较大波动,建议把单次定位超时时间设置为 10 秒。


以快速定位策略(PRIORITY_LOCATING_SPEED)为例,调用方式如下:


import { geoLocationManager } from '@kit.LocationKit';import { BusinessError } from '@kit.BasicServicesKit'let request: geoLocationManager.SingleLocationRequest = {   'locatingPriority': geoLocationManager.LocatingPriority.PRIORITY_LOCATING_SPEED,   'locatingTimeoutMs': 10000}try {   geoLocationManager.getCurrentLocation(request).then((result) => { // 调用getCurrentLocation获取当前设备位置,通过promise接收上报的位置      console.info('current location: ' + JSON.stringify(result));   })   .catch((error:BusinessError) => { // 接收上报的错误码      console.error('promise, getCurrentLocation: error=' + JSON.stringify(error));   }); } catch (err) {   console.error("errCode:" + JSON.stringify(err)); }
复制代码


通过本模块获取到的坐标均为 WGS-84 坐标系坐标点,如需使用其它坐标系类型的坐标点,请进行坐标系转换后再使用。


可参考 Map Kit 提供的地图计算工具进行坐标转换。


  1. 持续定位。多用于导航、运动轨迹、出行等场景。


首先要实例化 ContinuousLocationRequest 对象,用于告知系统该向应用提供何种类型的位置服务,以及位置结果上报的频率。


设置 locationScenario


建议 locationScenario 参数优先根据应用的使用场景进行设置,该参数枚举值定义参见 UserActivityScenario,例如地图在导航时使用 NAVIGATION 参数,可以持续在室内和室外场景获取位置用于导航。


设置 interval


表示上报位置信息的时间间隔,单位是秒,默认值为 1 秒。如果对位置上报时间间隔无特殊要求,可以不填写该字段。


以地图导航场景为例,调用方式如下:


import { geoLocationManager } from '@kit.LocationKit';let request: geoLocationManager.ContinuousLocationRequest= {   'interval': 1,   'locationScenario': geoLocationManager.UserActivityScenario.NAVIGATION}let locationCallback = (location:geoLocationManager.Location):void => {   console.info('locationCallback: data: ' + JSON.stringify(location));};try {   geoLocationManager.on('locationChange', request, locationCallback);} catch (err) {   console.error("errCode:" + JSON.stringify(err));}
复制代码


如果不主动结束定位可能导致设备功耗高,耗电快;建议在不需要获取定位信息时及时结束定位。


// 该回调函数需要与on接口传入的回调函数保持一致。geoLocationManager.off('locationChange', locationCallback);
复制代码


了解更多详情>>


访问位置服务联盟官网


获取位置服务开发指导文档

用户头像

HarmonyOS SDK

关注

HarmonyOS SDK 2022-06-16 加入

HarmonyOS SDK通过将HarmonyOS系统级能力对外开放,支撑开发者高效打造更纯净、更智能、更精致、更易用的鸿蒙应用,和开发者共同成长。

评论

发布
暂无评论
融合多元定位技术,帮助应用破解精准定位难题_HarmonyOS NEXT_HarmonyOS SDK_InfoQ写作社区