文章目录

  • 一、定位介绍
  • 二、代码
  • 1、初始化LocationManager
  • 2、处理运行时权限
  • 3、开始请求定位
  • 4、定位监听器
  • 5、卫星监听器
  • 6、更新定位
  • 三、遇到的坑
  • 四、Demo下载

一、定位介绍

在不使用第三方地图SDK的情况下,也可以借助Android
本身的 location api 来实现获取地理位置相关信息功能,官方介绍见这里:https://developer.android.com/reference/android/location/LocationManager

废话不多说,先看最终效果图:

android 代码获取经纬度 安卓手机查经纬度_定位

二、代码

1、初始化LocationManager

locationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE);

        if (!(locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER)
                || locationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER))) {
            Toast.makeText(this, "请打开网络或GPS定位功能!", Toast.LENGTH_SHORT).show();
            Intent intent = new Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS);
            startActivityForResult(intent, 0);
            return;
        }

2、处理运行时权限

if (ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED && ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED && ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
            ActivityCompat.requestPermissions(MainActivity.this,
                    new String[]{
                            Manifest.permission.ACCESS_COARSE_LOCATION,
                            Manifest.permission.ACCESS_FINE_LOCATION,
                            Manifest.permission.WRITE_EXTERNAL_STORAGE
                    }, 100);
        } else {
            initLocation();
        }

回调函数处理

@Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        switch (requestCode) {
            // 相机权限
            case 100:
                if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                    //用户点击了同意授权
                    initLocation();
                } else {
                    //用户拒绝了授权
                    Toast.makeText(MainActivity.this, "权限被拒绝", Toast.LENGTH_SHORT).show();
                }
                break;
        }
    }

3、开始请求定位

优先获取gps定位结果,没有的话再获取网络定位结果,并添加2个listener

long minSecond = 2 * 1000;
 		long minDistance = 2;
       Location location = locationManager.getLastKnownLocation(LocationManager.GPS_PROVIDER);
            if (location == null) {
                location = locationManager.getLastKnownLocation(LocationManager.NETWORK_PROVIDER);
            }
            updateLocation(location);

            locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, minSecond, minDistance, listener);//LocationManager.GPS_PROVIDER

            mGPSStatusImpl = new GPSStatusImpl();
            locationManager.addGpsStatusListener(mGPSStatusImpl);

4、定位监听器

public final LocationListener listener = new LocationListener() {
        @Override
        public void onLocationChanged(Location location) {
            updateLocation(location);
        }

        @Override
        public void onStatusChanged(String provider, int status, Bundle extras) {
            switch (status) {
                // GPS状态为可见时
                case LocationProvider.AVAILABLE:
                    Log.i("xml", "当前GPS状态为可见状态");
                    break;
                // GPS状态为服务区外时
                case LocationProvider.OUT_OF_SERVICE:
                    Log.i("xml", "当前GPS状态为服务区外状态");
                    break;
                // GPS状态为暂停服务时
                case LocationProvider.TEMPORARILY_UNAVAILABLE:
                    Log.i("xml", "当前GPS状态为暂停服务状态");
                    break;
            }
        }

        @Override
        public void onProviderEnabled(String provider) {
  			@SuppressLint("MissingPermission")
            Location location = locationManager.getLastKnownLocation(provider);
            updateLocation(location);
        }

        @Override
        public void onProviderDisabled(String provider) {
        }
    };

5、卫星监听器

在requestLocationUpdates方法后add监听器,在onDestroy或者onStop方法中remove该监听器

private class GPSStatusImpl implements GpsStatus.Listener
    {
        @Override
        public void onGpsStatusChanged(int event)
        {
            switch (event)
            {
                case GpsStatus.GPS_EVENT_FIRST_FIX:  //第一次定位
                    break;

                case GpsStatus.GPS_EVENT_SATELLITE_STATUS:  //卫星状态改变
                {
                    //获取当前状态
                    @SuppressLint("MissingPermission")
                    GpsStatus gpsStatus = locationManager.getGpsStatus(null);
                    //获取卫星颗数的默认最大值
                    int maxSatellites = gpsStatus.getMaxSatellites();
                    //创建一个迭代器保存所有卫星
                    Iterator<GpsSatellite> iters = gpsStatus.getSatellites().iterator();
                    int count = 0;
                    while (iters.hasNext() && count <= maxSatellites)
                    {
                        GpsSatellite s = iters.next();
                        // 只有信噪比不为0时才算合格的卫星
                        if (s.getSnr() != 0) {
                            count++;
                        }
                    }

                    stars = count;
			  		SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
                    String curTime = dateFormat.format(new Date());
                    System.out.println(curTime + "-----卫星数量nun----" + stars);
                    textView2.setText("实时卫星数量:" + stars);
                    break;
                }

                case GpsStatus.GPS_EVENT_STARTED:   //定位启动
                    break;

                case GpsStatus.GPS_EVENT_STOPPED:   //定位结束
                    break;
            }
        }
    }

6、更新定位

关键核心代码,把结果展示在textview并写入本地文件中

private void updateLocation(Location location) {
        String result;
        double lat;
        double lon;

        if (location != null) {
            lat = location.getLatitude();
            lon = location.getLongitude();
            if (location.hasSpeed()) {
                speed = location.getSpeed() * 3.6;
            } else {
                if( lastLocation != null) {
                    speed = getMySpeed(lastLocation, location);
                }
            }

            result = "纬度:" + lat + "\n经度:" + lon + "\n速度:" + String.format("%.2f", speed) + " km/h\n卫星数量:" + stars;
            Log.i("xml", "显示结果 = " +  result);
            lastLocation = location;
        } else {
            result = "无法获取经纬度信息";
        }

        SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        SimpleDateFormat dateFormatDate = new SimpleDateFormat("yyyyMMdd");
        String curTime = dateFormat.format(new Date());
        String curDate = dateFormatDate.format(new Date());

        // 把结果展示在界面上
        textView.setText(curTime + "\n" + result);

        // 把结果写入文件
        Utils.writeFileAppend(Utils.BASE_PROJECT_IMAGE_PATH + "log_" + curDate + ".txt", curTime + "\n" + result.replace("\n", " "));
    }

三、遇到的坑

使用LocationManager需要注意以下几个点:
(1)在开启网络定位的前提下,经过测试,如果只根据时间来定位的话,发现允许的最短时间是20秒,代码中设置低于这个时间的话,还是会以20秒作为定位间隔,与此同时,最小距离还必须设置为0,如果非0的话监听器就无法再被回调来获取网络定位经纬度。
(2)获取卫星数量时(简称搜星),尽量要在户外空旷的地方,同时搜星一般比较慢,而且代码中要实现GpsStatus.Listener接口,不然数量可能为0,同时要添加信噪比判断,不然容易获取较多的卫星数量
(3)在Manifest添加了相关权限并且代码中已经做了运行时权限处理代码,主要是以下两个权限:

Manifest.permission.ACCESS_COARSE_LOCATION
 Manifest.permission.ACCESS_FINE_LOCATION

如果requestLocationUpdates这个方法还是报红的话,那么只有添加注解来解决了

@SuppressLint("MissingPermission")

(4)在获取gps速度的时候,getSpeed方法要乘以3.6,这是为了把单位米每秒转换成千米每小时
(5)由于网络定位误差较大,建议只开启GPS定位,当hasSpeed为false的时候,使用两个经纬度之差和时间之差来获取实时速度