Android 中获取LocationProvider的三种方法和获取定位信息

简介: Android 中获取LocationProvider的三种方法和获取定位信息

前言:

LocationProvider是位置源的意思,用于提供定位信息。

常用的LocationProvider主要有三种:

  1. GPS:通过手机里面的GPS芯片,来利用卫星定位信息的。
  2. network:通过网络来获取位置信息的,主要利用手机的基站,和WiFi节点的位置来大致定位。
  3. passive:是一种被动定位方式,它自己不能获取定位信息,而是利用被系统保存的其他程序所更新的定位信息。

下面我们通过三种方法获取LocationProvider

一、首先是activity_location_provider.xml布局文件,代码如下

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".LocationProviderActivity">
    <TextView
        android:id="@+id/textView"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:gravity="center"
        android:text="可用的LocationProvider(位置提供者)"
        android:textColor="@color/black"
        android:textSize="20sp" />
    <TextView
        android:id="@+id/text_locationProvider"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:gravity="center"
        android:textSize="20sp" />
</LinearLayout>

二、接着在LocationProviderActivity类中获取LocationProvider,具体讲解已经在代码中给出

public class LocationProviderActivity extends AppCompatActivity {
    private TextView text_locationProvider;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_location_provider);
        text_locationProvider = findViewById(R.id.text_locationProvider);
        if (Build.VERSION.SDK_INT>Build.VERSION_CODES.M){
            if (ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION)!= PackageManager.PERMISSION_GRANTED){
                ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.ACCESS_FINE_LOCATION},1);
            }
        }
        LocationManager locationManager = (LocationManager) getSystemService(LOCATION_SERVICE);
        //第一种方式:获取所有的LocationProvider名称,通过LocationManager对象的getAllProviders()来实现
//        List<String> providerNames= locationManager.getAllProviders();
//        StringBuilder stringBuilder = new StringBuilder();
//        Iterator<String> iterator = providerNames.iterator();
//        while (iterator.hasNext()){
//            stringBuilder.append(iterator.next()+"\n");
//        }
//        //显示获取的locationProvider名称
//        text_locationProvider.setText(stringBuilder.toString());
        //第二种方法:通过名称获得LocationProvider
        //获取基于PASSIVE的locationProvider
//        LocationProvider locationProvider = locationManager.getProvider(LocationManager.PASSIVE_PROVIDER);
//        //获取LocationProvider名称
//        text_locationProvider.setText(locationProvider.getName());
        //三、通过Criteria类获得LocationProvider
        //获取最佳的LocationProvider
        //创建一个过滤条件对象
        Criteria criteria = new Criteria();
        //使用不收费的
        criteria.setCostAllowed(false);
        //使用精度最准确的
        criteria.setAccuracy(Criteria.ACCURACY_FINE);
        //使用耗电量最低的
        criteria.setPowerRequirement(Criteria.POWER_LOW);
        //获取最佳的LocationProvider名称 第二个参数 表示是可用的
        String provider = locationManager.getBestProvider(criteria,true);
        text_locationProvider.setText(provider);
    }
}

效果如图:

我这是虚拟机运行,所以只有这一种方式,正常真机运行便是三种方式。

下面通过LocationProvider获取定位信息,分别为经度和纬度,通过经度和纬度便可得到所在的位置信息。

首先是布局activity_location.xml,一个文本框显示经度和纬度的这里就不再给出代码

主要看LocationActivity类中实现的代码,具体讲解也已经给出了注释:

public class LocationActivity extends AppCompatActivity {
    private TextView textView;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_location);
        textView = findViewById(R.id.textView);
        //获取位置管理器
        LocationManager locationManager = (LocationManager) getSystemService(LOCATION_SERVICE);
        //申请权限
        if (ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED && ContextCompat.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}, 1);
        }
        //这个监听器实现每隔1s中更新一次位置信息
        locationManager.requestLocationUpdates(
                LocationManager.GPS_PROVIDER,//指定GPS定位的提供者
                1000,//间隔时间
                1,//位置更新之间的最小距离
                new LocationListener() { //监听GPS定位信息是否改变
                    @Override
                    public void onLocationChanged(@NonNull Location location) { //GPS信息发生改变时回调
                    }
                }
        );
        //获取最新的定位信息
        Location location = locationManager.getLastKnownLocation(LocationManager.GPS_PROVIDER);
        //将最新的定位信息,传递给LocationUpdates()方法
        locationUpdates(location);
    }
    public void locationUpdates(Location location) {
        if (null != location) {
            StringBuilder stringBuilder = new StringBuilder();
            stringBuilder.append("您的位置是:\n");
            stringBuilder.append("经度:");
            stringBuilder.append(location.getLongitude());
            stringBuilder.append("\n纬度:");
            stringBuilder.append(location.getLatitude());
            textView.setText(stringBuilder.toString());
        } else {
            textView.setText("没有获取到GPS信息");
        }
    }
}

具体效果如图所示:

通过纬度(latitude),经度(longitude)来获取具体信息,代码紧接着上面的演示

public void locationUpdates(Location location) {
        if (null != location) {
            StringBuilder stringBuilder = new StringBuilder();
            stringBuilder.append("您的位置是:\n");
            stringBuilder.append("经度:");
            stringBuilder.append(location.getLongitude());
            stringBuilder.append("\n纬度:");
            stringBuilder.append(location.getLatitude());
            textView.setText(stringBuilder.toString());
            getAddress(location.getLongitude(), location.getLatitude());
        } else {
            textView.setText("没有获取到GPS信息");
        }
    }
    private void getAddress(double longitude, double latitude) {
        //Geocoder通过经纬度获取具体信息
        Geocoder gc = new Geocoder(this, Locale.getDefault());
        try {
            List<Address> addressList = gc.getFromLocation(latitude, longitude, 1);
            if (addressList != null) {
                Address address = addressList.get(0);
                //获取国家名称
                String countryName = address.getCountryName();
                //返回地址的国家代码,CN
                String countryCode = address.getCountryCode();
                Log.d("TAG", "getAddress: "+countryCode);
                //对应的省或者市
                String adminArea = address.getAdminArea();
                //子管理区域 对应的镇
                String subAdminArea = address.getSubAdminArea();
                //一个市对应的具体的区
                String subLocality = address.getSubLocality();
                //具体镇名加具体位置
                String featureName = address.getFeatureName();
                //返回一个具体的位置串,这个就不用进行拼接了。
                String addressLines =address.getAddressLine(0);
                String specificAddress = countryName + adminArea + subLocality + featureName;
                text_address.setText(addressLines);
//                text_address.setText(specificAddress);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

具体注释已经在代码中给出,以上就是获取定位信息的具体方法~


目录
相关文章
|
7月前
|
Android开发 开发者
Android自定义view之利用drawArc方法实现动态效果
本文介绍了如何通过Android自定义View实现动态效果,重点使用`drawArc`方法完成圆弧动画。首先通过`onSizeChanged`进行测量,初始化画笔属性,设置圆弧相关参数。核心思路是不断改变圆弧扫过角度`sweepAngle`,并调用`invalidate()`刷新View以实现动态旋转效果。最后附上完整代码与效果图,帮助开发者快速理解并实践这一动画实现方式。
180 0
|
5月前
|
安全 数据库 Android开发
在Android开发中实现两个Intent跳转及数据交换的方法
总结上述内容,在Android开发中,Intent不仅是活动跳转的桥梁,也是两个活动之间进行数据交换的媒介。运用Intent传递数据时需注意数据类型、传输大小限制以及安全性问题的处理,以确保应用的健壯性和安全性。
364 11
|
存储 Shell Android开发
基于Android P,自定义Android开机动画的方法
本文详细介绍了基于Android P系统自定义开机动画的步骤,包括动画文件结构、脚本编写、ZIP打包方法以及如何将自定义动画集成到AOSP源码中。
569 2
基于Android P,自定义Android开机动画的方法
|
Android开发
基于android-11.0.0_r39,系统应用的手动签名方法和过程
本文介绍了基于Android 11.0.0_r39版本进行系统应用手动签名的方法和解决签名过程中遇到的错误,包括处理`no conscrypt_openjdk_jni-linux-x86_64`和`RegisterNatives failed`的问题。
667 2
|
缓存 Java Shell
Android 系统缓存扫描与清理方法分析
Android 系统缓存从原理探索到实现。
489 15
Android 系统缓存扫描与清理方法分析
|
Java 测试技术 Android开发
Android性能测试——发现和定位内存泄露和卡顿
本文详细介绍了Android应用性能测试中的内存泄漏与卡顿问题及其解决方案。首先,文章描述了使用MAT工具定位内存泄漏的具体步骤,并通过实例展示了如何分析Histogram图表和Dominator Tree。接着,针对卡顿问题,文章探讨了其产生原因,并提供了多种测试方法,包括GPU呈现模式分析、FPS Meter软件测试、绘制圆点计数法及Android Studio自带的GPU监控功能。最后,文章给出了排查卡顿问题的四个方向,帮助开发者优化应用性能。
1016 4
Android性能测试——发现和定位内存泄露和卡顿
|
Java Unix Linux
Android Studio中Terminal运行./gradlew clean build提示错误信息
遇到 `./gradlew clean build`命令执行出错时,首先应检查错误信息的具体内容,这通常会指向问题的根源。从权限、环境配置、依赖下载、版本兼容性到项目配置本身,逐一排查并应用相应的解决措施。记住,保持耐心,逐步解决问题,往往复杂问题都是由简单原因引起的。
1141 2
|
ARouter 测试技术 API
Android经典面试题之组件化原理、优缺点、实现方法?
本文介绍了组件化在Android开发中的应用,详细阐述了其原理、优缺点及实现方式,包括模块化、接口编程、依赖注入、路由机制等内容,并提供了具体代码示例。
295 2
|
Android开发
Android在rootdir根目录创建自定义目录和挂载点的方法
本文介绍了在Android高通平台的根目录下创建自定义目录和挂载点的方法,通过修改Android.mk文件并使用`LOCAL_POST_INSTALL_CMD`变量在编译过程中添加目录,最终在ramdisk.img的系统根路径下成功创建了`/factory/bin`目录。
725 1
|
开发工具 uml git
AOSP源码下载方法,解决repo sync错误:android-13.0.0_r82
本文分享了下载AOSP源码的方法,包括如何使用repo工具和处理常见的repo sync错误,以及配置Python环境以确保顺利同步特定版本的AOSP代码。
2374 0
AOSP源码下载方法,解决repo sync错误:android-13.0.0_r82