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

Android通过原生方式获取经纬度与城市信息的方法

这篇文章主要给大家介绍了关于Android通过原生方式获取经纬度与城市信息的相关资料,文中通过示例代码介绍的非常详细,对各位Android开发者们具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧

一、概述

在项目中需要获取用户所在位置的经纬度和城市上送给风控系统。一般来说,定位有两种方式:

  1. 用第三方SDK定位,如百度地图、高德地图、谷歌地图;
  2. 用Android原生SDK中的api定位;

本文讲述定位的第二种方式--用Android原生的SDK中的api定位,如果项目定位要求较高还是建议使用第三方地图库。

二、Android原生SDK中的api定位

Android原生方式获取经纬度两种定位方式:GPS定位和Wifi定位

  • GPS定位相比Wifi定位更精准且可在无网络情况下使用,但在室内基本暴毙无法使用。
  • WiFi定位没有室内外限制,也不需要开启GPS但需要联网。但测试发现WiFi定位时onLocationChanged函数(用于监听经纬度变化)触发间隔无法小于30s。

示例代码如下:

public class TestLocationActivity extends AppCompatActivity {

 public static final int LOCATION_CODE = 301;
 private LocationManager locationManager;
 private String locatiOnProvider= null;


 @Override
 protected void onCreate(@Nullable Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  getLocation();
 }

 private void getLocation(){
  //1.获取位置管理器
  locatiOnManager= (LocationManager) getSystemService(Context.LOCATION_SERVICE);

  //2.获取位置提供器,GPS或是NetWork
  List providers = locationManager.getProviders(true);

  if (providers.contains(LocationManager.GPS_PROVIDER)) {
   //如果是GPS
   locatiOnProvider= LocationManager.GPS_PROVIDER;
   Log.v("TAG", "定位方式GPS");
  } else if (providers.contains(LocationManager.NETWORK_PROVIDER)) {
   //如果是Network
   locatiOnProvider= LocationManager.NETWORK_PROVIDER;
   Log.v("TAG", "定位方式Network");
  }else {
   Toast.makeText(this, "没有可用的位置提供器", Toast.LENGTH_SHORT).show();
   return;
  }

  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
   //获取权限(如果没有开启权限,会弹出对话框,询问是否开启权限)
   if (ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) 
     != PackageManager.PERMISSION_GRANTED || 
    ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_COARSE_LOCATION) 
     != PackageManager.PERMISSION_GRANTED) {
    //请求权限
    ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.ACCESS_FINE_LOCATION,
      Manifest.permission.ACCESS_COARSE_LOCATION}, LOCATION_CODE);
   } else {
    //3.获取上次的位置,一般第一次运行,此值为null
    Location location = locationManager.getLastKnownLocation(locationProvider);
    if (location!=null){
     Toast.makeText(this, location.getLongitude() + " " + 
           location.getLatitude() + "",Toast.LENGTH_SHORT).show();
     Log.v("TAG", "获取上次的位置-经纬度:"+location.getLongitude()+" "+location.getLatitude());
     getAddress(location);

    }else{
     //监视地理位置变化,第二个和第三个参数分别为更新的最短时间minTime和最短距离minDistace
     locationManager.requestLocationUpdates(locationProvider, 3000, 1,locationListener);
    }
   }
  } else {
   Location location = locationManager.getLastKnownLocation(locationProvider);
   if (location!=null){
    Toast.makeText(this, location.getLongitude() + " " + 
          location.getLatitude() + "", Toast.LENGTH_SHORT).show();
    Log.v("TAG", "获取上次的位置-经纬度:"+location.getLongitude()+" "+location.getLatitude());
    getAddress(location);

   }else{
    //监视地理位置变化,第二个和第三个参数分别为更新的最短时间minTime和最短距离minDistace
    locationManager.requestLocationUpdates(locationProvider, 3000, 1,locationListener);
   }
  }
 }

 public LocationListener locatiOnListener= new LocationListener() {
  // Provider的状态在可用、暂时不可用和无服务三个状态直接切换时触发此函数
  @Override
  public void onStatusChanged(String provider, int status, Bundle extras) {
  }
  // Provider被enable时触发此函数,比如GPS被打开
  @Override
  public void onProviderEnabled(String provider) {
  }
  // Provider被disable时触发此函数,比如GPS被关闭
  @Override
  public void onProviderDisabled(String provider) {
  }
  //当坐标改变时触发此函数,如果Provider传进相同的坐标,它就不会被触发
  @Override
  public void onLocationChanged(Location location) {
   if (location != null) {
    //如果位置发生变化,重新显示地理位置经纬度
    Toast.makeText(TestLocationActivity.this, location.getLongitude() + " " + 
               location.getLatitude() + "", Toast.LENGTH_SHORT).show();
    Log.v("TAG", "监视地理位置变化-经纬度:"+location.getLongitude()+" "+location.getLatitude());
   }
  }
 };

 @Override
 public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
  switch (requestCode) {
   case LOCATION_CODE:
    if(grantResults.length > 0 && grantResults[0] == getPackageManager().PERMISSION_GRANTED
      && grantResults[1] == PackageManager.PERMISSION_GRANTED) {
     Toast.makeText(this, "申请权限", Toast.LENGTH_LONG).show();
     try {
      List providers = locationManager.getProviders(true);
      if (providers.contains(LocationManager.NETWORK_PROVIDER)) {
       //如果是Network
       locatiOnProvider= LocationManager.NETWORK_PROVIDER;

      }else if (providers.contains(LocationManager.GPS_PROVIDER)) {
       //如果是GPS
       locatiOnProvider= LocationManager.GPS_PROVIDER;
      }
      Location location = locationManager.getLastKnownLocation(locationProvider);
      if (location!=null){
       Toast.makeText(this, location.getLongitude() + " " + 
             location.getLatitude() + "", Toast.LENGTH_SHORT).show();
       Log.v("TAG", "获取上次的位置-经纬度:"+location.getLongitude()+" "+location.getLatitude());
      }else{
       // 监视地理位置变化,第二个和第三个参数分别为更新的最短时间minTime和最短距离minDistace
       locationManager.requestLocationUpdates(locationProvider, 0, 0,locationListener);
      }

     }catch (SecurityException e){
      e.printStackTrace();
     }
    } else {
     Toast.makeText(this, "缺少权限", Toast.LENGTH_LONG).show();
     finish();
    }
    break;
  }
 }


 //获取地址信息:城市、街道等信息
 private List
getAddress(Location location) { List
result = null; try { if (location != null) { Geocoder gc = new Geocoder(this, Locale.getDefault()); result = gc.getFromLocation(location.getLatitude(), location.getLongitude(), 1); Toast.makeText(this, "获取地址信息:"+result.toString(), Toast.LENGTH_LONG).show(); Log.v("TAG", "获取地址信息:"+result.toString()); } } catch (Exception e) { e.printStackTrace(); } return result; } @Override protected void onDestroy() { super.onDestroy(); locationManager.removeUpdates(locationListener); } }

在AndroidManifest.xml加权限


 


三、总结

首先将手机定位服务设置,调到下图所示:

这可能会获取不到经纬度。为什么在网络和GPS都可用的情况下只执行GPS而不是网络?也许是从精确度考虑的,但是走GPS进入监听的listener后,不会执行onLocationChanged()方法,因为我是在室内,没有移动,所以如果你获取不到经纬度,就要将定位服务调到仅使用网络定位或者关闭手机GPS这样就可以获取到了。

总结

到此这篇关于Android通过原生方式获取经纬度与城市信息的文章就介绍到这了,更多相关Android原生获取经纬度与城市信息内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!


推荐阅读
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • YOLOv7基于自己的数据集从零构建模型完整训练、推理计算超详细教程
    本文介绍了关于人工智能、神经网络和深度学习的知识点,并提供了YOLOv7基于自己的数据集从零构建模型完整训练、推理计算的详细教程。文章还提到了郑州最低生活保障的话题。对于从事目标检测任务的人来说,YOLO是一个熟悉的模型。文章还提到了yolov4和yolov6的相关内容,以及选择模型的优化思路。 ... [详细]
  • 本文介绍了使用kotlin实现动画效果的方法,包括上下移动、放大缩小、旋转等功能。通过代码示例演示了如何使用ObjectAnimator和AnimatorSet来实现动画效果,并提供了实现抖动效果的代码。同时还介绍了如何使用translationY和translationX来实现上下和左右移动的效果。最后还提供了一个anim_small.xml文件的代码示例,可以用来实现放大缩小的效果。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文讲述了如何通过代码在Android中更改Recycler视图项的背景颜色。通过在onBindViewHolder方法中设置条件判断,可以实现根据条件改变背景颜色的效果。同时,还介绍了如何修改底部边框颜色以及提供了RecyclerView Fragment layout.xml和项目布局文件的示例代码。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • baresip android编译、运行教程1语音通话
    本文介绍了如何在安卓平台上编译和运行baresip android,包括下载相关的sdk和ndk,修改ndk路径和输出目录,以及创建一个c++的安卓工程并将目录考到cpp下。详细步骤可参考给出的链接和文档。 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • 安卓select模态框样式改变_微软Office风格的多端(Web、安卓、iOS)组件库——Fabric UI...
    介绍FabricUI是微软开源的一套Office风格的多端组件库,共有三套针对性的组件,分别适用于web、android以及iOS,Fab ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • 本文是关于自学Android的笔记,包括查看类的源码的方法,活动注册的必要性以及布局练习的重要性。通过学习本文,读者可以了解到在自学Android过程中的一些关键点和注意事项。 ... [详细]
author-avatar
大侠aaaaaaaaaaa_225
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有