Android通过原生APi获取所在位置的经纬度

2024-05-02 23:32

本文主要是介绍Android通过原生APi获取所在位置的经纬度,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在Android开发当中,经常需要用到定位功能,尤其是依赖于地理位置功能的应用,本文介绍了Android通过原生APi获取所在位置的经纬度,分享给大家

一、难点介绍

1.难点

我们的应用要新增一个功能,就是在用户打开附件的人页面后,将用户的经纬度通过一个接口返回给服务端,从而让服务器可以准确定位。

因为只是添加一个小功能所以,引入第三方SDK定位就有些大材小用了,所以就准备借助原生APi:LocationManager来完成。
经过在网络上一搜,有很多关于通过LocationManager获取经纬度坐标的,所有的代码几乎大同小异,本着负责任的态度,就看了好多篇,然后总结出了一个比较优良的获取经纬度的工具类。

在验证的过程中遇到了以下的几个问题:

①权限问题:Android 6.0之后新增动态权限,而获取获取经纬度坐标的权限如果你的app的(TargetVersion>=23)就需要动态获取了(当然也得在清单文件中设置)

②谷歌网络服务在中国被禁,所以就会导致网络定位在中国不可用, 只能使用GPS定位,而GPS定位需要用户打开GPS标志才能获取到

其中最让我难以解决的是第二个问题。因为尝试了很多次,只能通过打开GPS才能获取到位置。终于在问遍各位大神后,都说了关于是因为谷歌网络服务在中国被禁。

所幸,我们应用面对的群体是海外用户,所以不存在谷歌服务被墙的情况。

以上就是我在过程中遇到的问题了,如果解决了你的困惑的话,那就太开心了。

关于Android6.0动态权限的话,只要写好对应的回调就可以了,不算很难。

下面贴上我工具类的代码给大家:

public class LocationUtils {private volatile static LocationUtils uniqueInstance;private LocationManager locationManager;private String locationProvider;private Location location;private Context mContext;private LocationUtils(Context context) {mContext = context;getLocation();}//采用Double CheckLock(DCL)实现单例public static LocationUtils getInstance(Context context) {if (uniqueInstance == null) {synchronized (LocationUtils.class) {if (uniqueInstance == null) {uniqueInstance = new LocationUtils( context );}}}return uniqueInstance;}private void getLocation() {//1.获取位置管理器locationManager = (LocationManager) mContext.getSystemService( Context.LOCATION_SERVICE );//2.获取位置提供器,GPS或是NetWorkList<String> providers = locationManager.getProviders( true );if (providers.contains( LocationManager.NETWORK_PROVIDER )) {//如果是网络定位Log.d( TAG, "如果是网络定位" );locationProvider = LocationManager.NETWORK_PROVIDER;} else if (providers.contains( LocationManager.GPS_PROVIDER )) {//如果是GPS定位Log.d( TAG, "如果是GPS定位" );locationProvider = LocationManager.GPS_PROVIDER;} else {Log.d( TAG, "没有可用的位置提供器" );return;}// 需要检查权限,否则编译报错,想抽取成方法都不行,还是会报错。只能这样重复 code 了。if (Build.VERSION.SDK_INT >= 23 &&ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_FINE_LOCATION ) != PackageManager.PERMISSION_GRANTED &&ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_COARSE_LOCATION ) != PackageManager.PERMISSION_GRANTED) {return;}if (ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_FINE_LOCATION ) != PackageManager.PERMISSION_GRANTED && ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_COARSE_LOCATION ) != PackageManager.PERMISSION_GRANTED) {return;}//3.获取上次的位置,一般第一次运行,此值为nullLocation location = locationManager.getLastKnownLocation( locationProvider );if (location != null) {setLocation( location );}// 监视地理位置变化,第二个和第三个参数分别为更新的最短时间minTime和最短距离minDistacelocationManager.requestLocationUpdates( locationProvider, 0, 0, locationListener );}private void setLocation(Location location) {this.location = location;String address = "纬度:" + location.getLatitude() + "经度:" + location.getLongitude();Log.d( TAG, address );}//获取经纬度public Location showLocation() {return location;}// 移除定位监听public void removeLocationUpdatesListener() {// 需要检查权限,否则编译不过if (Build.VERSION.SDK_INT >= 23 &&ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_FINE_LOCATION ) != PackageManager.PERMISSION_GRANTED &&ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_COARSE_LOCATION ) != PackageManager.PERMISSION_GRANTED) {return;}if (locationManager != null) {uniqueInstance = null;locationManager.removeUpdates( locationListener );}}/*** LocationListern监听器* 参数:地理位置提供器、监听位置变化的时间间隔、位置变化的距离间隔、LocationListener监听器*/LocationListener locationListener = new LocationListener() {/*** 当某个位置提供者的状态发生改变时*/@Overridepublic void onStatusChanged(String provider, int status, Bundle arg2) {}/*** 某个设备打开时*/@Overridepublic void onProviderEnabled(String provider) {}/*** 某个设备关闭时*/@Overridepublic void onProviderDisabled(String provider) {}/*** 手机位置发生变动*/@Overridepublic void onLocationChanged(Location location) {location.getAccuracy();//精确度setLocation( location );}};}

用法:

public class MainActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate( savedInstanceState );setContentView( R.layout.activity_main );Button btn = (Button) findViewById( R.id.btn );final TextView text = (TextView) findViewById( R.id.text );btn.setOnClickListener( new View.OnClickListener() {@Overridepublic void onClick(View v) {Location location = LocationUtils.getInstance( MainActivity.this ).showLocation();if (location != null) {String address = "纬度:" + location.getLatitude() + "经度:" + location.getLongitude();Log.d( "FLY.LocationUtils", address );text.setText( address );}}} );}@Overrideprotected void onDestroy() {super.onDestroy();LocationUtils.getInstance( this ).removeLocationUpdatesListener();}
}
更多信息,详见  https://www.cnblogs.com/kdkler/p/6196751.html

这篇关于Android通过原生APi获取所在位置的经纬度的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/955366

相关文章

SpringBoot监控API请求耗时的6中解决解决方案

《SpringBoot监控API请求耗时的6中解决解决方案》本文介绍SpringBoot中记录API请求耗时的6种方案,包括手动埋点、AOP切面、拦截器、Filter、事件监听、Micrometer+... 目录1. 简介2.实战案例2.1 手动记录2.2 自定义AOP记录2.3 拦截器技术2.4 使用Fi

Python获取浏览器Cookies的四种方式小结

《Python获取浏览器Cookies的四种方式小结》在进行Web应用程序测试和开发时,获取浏览器Cookies是一项重要任务,本文我们介绍四种用Python获取浏览器Cookies的方式,具有一定的... 目录什么是 Cookie?1.使用Selenium库获取浏览器Cookies2.使用浏览器开发者工具

Java获取当前时间String类型和Date类型方式

《Java获取当前时间String类型和Date类型方式》:本文主要介绍Java获取当前时间String类型和Date类型方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录Java获取当前时间String和Date类型String类型和Date类型输出结果总结Java获取

C#监听txt文档获取新数据方式

《C#监听txt文档获取新数据方式》文章介绍通过监听txt文件获取最新数据,并实现开机自启动、禁用窗口关闭按钮、阻止Ctrl+C中断及防止程序退出等功能,代码整合于主函数中,供参考学习... 目录前言一、监听txt文档增加数据二、其他功能1. 设置开机自启动2. 禁止控制台窗口关闭按钮3. 阻止Ctrl +

一文详解如何使用Java获取PDF页面信息

《一文详解如何使用Java获取PDF页面信息》了解PDF页面属性是我们在处理文档、内容提取、打印设置或页面重组等任务时不可或缺的一环,下面我们就来看看如何使用Java语言获取这些信息吧... 目录引言一、安装和引入PDF处理库引入依赖二、获取 PDF 页数三、获取页面尺寸(宽高)四、获取页面旋转角度五、判断

SpringBoot3.X 整合 MinIO 存储原生方案

《SpringBoot3.X整合MinIO存储原生方案》本文详细介绍了SpringBoot3.X整合MinIO的原生方案,从环境搭建到核心功能实现,涵盖了文件上传、下载、删除等常用操作,并补充了... 目录SpringBoot3.X整合MinIO存储原生方案:从环境搭建到实战开发一、前言:为什么选择MinI

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

Knife4j+Axios+Redis前后端分离架构下的 API 管理与会话方案(最新推荐)

《Knife4j+Axios+Redis前后端分离架构下的API管理与会话方案(最新推荐)》本文主要介绍了Swagger与Knife4j的配置要点、前后端对接方法以及分布式Session实现原理,... 目录一、Swagger 与 Knife4j 的深度理解及配置要点Knife4j 配置关键要点1.Spri

Python使用OpenCV实现获取视频时长的小工具

《Python使用OpenCV实现获取视频时长的小工具》在处理视频数据时,获取视频的时长是一项常见且基础的需求,本文将详细介绍如何使用Python和OpenCV获取视频时长,并对每一行代码进行深入解析... 目录一、代码实现二、代码解析1. 导入 OpenCV 库2. 定义获取视频时长的函数3. 打开视频文