热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

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

在Android开发当中,经常需要用到定位功能,尤其是依赖于地理位置功能的应用,本文介绍了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或是NetWork
    List 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.获取上次的位置,一般第一次运行,此值为null
    Location location = locationManager.getLastKnownLocation( locationProvider );
    if (location != null) {
      setLocation( location );
    }
    // 监视地理位置变化,第二个和第三个参数分别为更新的最短时间minTime和最短距离minDistace
    locationManager.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() {

    /**
    * 当某个位置提供者的状态发生改变时
    */
    @Override
    public void onStatusChanged(String provider, int status, Bundle arg2) {

    }

    /**
    * 某个设备打开时
    */
    @Override
    public void onProviderEnabled(String provider) {

    }

    /**
    * 某个设备关闭时
    */
    @Override
    public void onProviderDisabled(String provider) {

    }

    /**
    * 手机位置发生变动
    */
    @Override
    public void onLocationChanged(Location location) {
      location.getAccuracy();//精确度
      setLocation( location );
    }
  };

}

用法:

public class MainActivity extends AppCompatActivity {

  @Override
  protected 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() {
      @Override
      public 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 );
        }

      }
    } );
  }

  @Override
  protected void onDestroy() {
    super.onDestroy();
    LocationUtils.getInstance( this ).removeLocationUpdatesListener();
  }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 深入理解Cookie与Session会话管理
    本文详细介绍了如何通过HTTP响应和请求处理浏览器的Cookie信息,以及如何创建、设置和管理Cookie。同时探讨了会话跟踪技术中的Session机制,解释其原理及应用场景。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 360SRC安全应急响应:从漏洞提交到修复的全过程
    本文详细介绍了360SRC平台处理一起关键安全事件的过程,涵盖从漏洞提交、验证、排查到最终修复的各个环节。通过这一案例,展示了360在安全应急响应方面的专业能力和严谨态度。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • 本文详细介绍如何使用arm-eabi-gdb调试Android平台上的C/C++程序。通过具体步骤和实用技巧,帮助开发者更高效地进行调试工作。 ... [详细]
  • 本文介绍了如何使用JQuery实现省市二级联动和表单验证。首先,通过change事件监听用户选择的省份,并动态加载对应的城市列表。其次,详细讲解了使用Validation插件进行表单验证的方法,包括内置规则、自定义规则及实时验证功能。 ... [详细]
  • 数据库内核开发入门 | 搭建研发环境的初步指南
    本课程将带你从零开始,逐步掌握数据库内核开发的基础知识和实践技能,重点介绍如何搭建OceanBase的开发环境。 ... [详细]
  • 本文详细介绍了如何使用 Yii2 的 GridView 组件在列表页面实现数据的直接编辑功能。通过具体的代码示例和步骤,帮助开发者快速掌握这一实用技巧。 ... [详细]
  • 解决MongoDB Compass远程连接问题
    本文记录了在使用阿里云服务器部署MongoDB后,通过MongoDB Compass进行远程连接时遇到的问题及解决方案。详细介绍了从防火墙配置到安全组设置的各个步骤,帮助读者顺利解决问题。 ... [详细]
  • 本文详细介绍如何使用Samba软件配置CIFS文件共享服务,涵盖安装、配置、权限管理及多用户挂载等关键步骤。通过具体示例和命令行操作,帮助读者快速搭建并优化Samba服务器。 ... [详细]
  • 本文详细探讨了HTTP 500内部服务器错误的成因、解决方案及其在Web开发中的影响。通过对具体案例的分析,帮助读者理解并解决此类问题。 ... [详细]
  • 本文介绍了如何使用PHP代码实现微信平台的媒体素材上传功能,详细解释了API接口的使用方法和注意事项,确保文件路径正确以避免常见的错误。 ... [详细]
  • 使用Vultr云服务器和Namesilo域名搭建个人网站
    本文详细介绍了如何通过Vultr云服务器和Namesilo域名搭建一个功能齐全的个人网站,包括购买、配置服务器以及绑定域名的具体步骤。文章还提供了详细的命令行操作指南,帮助读者顺利完成建站过程。 ... [详细]
  • 本文详细介绍了 MySQL 的查询处理流程,包括从客户端连接到服务器、查询缓存检查、语句解析、查询优化及执行等步骤。同时,深入探讨了 MySQL 中的乐观锁机制及其在并发控制中的应用。 ... [详细]
author-avatar
流浪的牛仔2011Ting_883
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有