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

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

概述在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<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.获取上次的位置,一般第一次运行,此值为null    Location location = locationManager.getLastKNownLocation( locationProvIDer );    if (location != null) {      setLocation( location );    }    // 监视地理位置变化,第二个和第三个参数分别为更新的最短时间minTime和最短距离mindistace    locationManager.requestLocationUpdates( locationProvIDer,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_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();  }}

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

总结

以上是内存溢出为你收集整理的Android通过原生APi获取所在位置的经纬度全部内容,希望文章能够帮你解决Android通过原生APi获取所在位置的经纬度所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/web/1145628.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-31
下一篇 2022-05-31

发表评论

登录后才能评论

评论列表(0条)

保存