Android 模擬地圖定位功能的實現

實現原理:

手機定位方式目前有4種:

  • 基站定位
  • WIFI定位
  • GPS定位
  • AGPS定位

本工程利用手機自帶的"模擬位置"功能實現運行時修改LocationManager結果。

原理:使用android自帶的調試api,模擬gps provider的結果。 

Android 6.0系統以下,可以通過Setting.Secure.ALLOW_MOCK_LOCATION獲取是否【允許模擬位置】,當【允許模擬位置】開啟時,可addTestProvider;

  Android 6.0系統及以上,棄用Setting.Secure.ALLOW_MOCK_LOCATION變量,沒有【允許模擬位置】選項,
增加【選擇模擬位置信息應用】,此時需要選擇當前應用,才可以addTestProvider,
但未找到獲取當前選擇應用的方法,因此通過addTestProvider是否成功來判斷是否可用模擬位置。

代碼分析:

MockLocationManager:模擬地址管理類
首先通過Android系統模擬位置管理器LocationManager獲取系統模擬位置服務,Android 6.0以下,通過Setting.Secure.ALLOW_MOCK_LOCATION判斷是否可模擬位置,Android 6.0及以上,需要【選擇模擬位置信息應用】,未找到方法,因此通過addTestProvider是否可用判斷。

    /**
     * 模擬位置是否啟用
     * 若啟用,則addTestProvider
     */
    public boolean getUseMockPosition(Context context) {
        // Android 6.0以下,通過Setting.Secure.ALLOW_MOCK_LOCATION判斷
        // Android 6.0及以上,需要【選擇模擬位置信息應用】,未找到方法,因此通過addTestProvider是否可用判斷
        boolean canMockPosition = (Settings.Secure.getInt(context.getContentResolver(), Settings.Secure.ALLOW_MOCK_LOCATION, 0) != 0)
                || Build.VERSION.SDK_INT > 22;
        if (canMockPosition && hasAddTestProvider == false) {
            try {
                for (String providerStr : mockProviders) {
                    LocationProvider provider = locationManager.getProvider(providerStr);
                    if (provider != null) {
                        locationManager.addTestProvider(
                                provider.getName()
                                , provider.requiresNetwork()
                                , provider.requiresSatellite()
                                , provider.requiresCell()
                                , provider.hasMonetaryCost()
                                , provider.supportsAltitude()
                                , provider.supportsSpeed()
                                , provider.supportsBearing()
                                , provider.getPowerRequirement()
                                , provider.getAccuracy());
                    } else {
                        if (providerStr.equals(LocationManager.GPS_PROVIDER)) {
                            locationManager.addTestProvider(
                                    providerStr
                                    , true, true, false, false, true, true, true
                                    , Criteria.POWER_HIGH, Criteria.ACCURACY_FINE);
                        } else if (providerStr.equals(LocationManager.NETWORK_PROVIDER)) {
                            locationManager.addTestProvider(
                                    providerStr
                                    , true, false, true, false, false, false, false
                                    , Criteria.POWER_LOW, Criteria.ACCURACY_FINE);
                        } else {
                            locationManager.addTestProvider(
                                    providerStr
                                    , false, false, false, false, true, true, true
                                    , Criteria.POWER_LOW, Criteria.ACCURACY_FINE);
                        }
                    }
                    locationManager.setTestProviderEnabled(providerStr, true);
                    locationManager.setTestProviderStatus(providerStr, LocationProvider.AVAILABLE, null, System.currentTimeMillis());
                }
                hasAddTestProvider = true;  // 模擬位置可用
                canMockPosition = true;
            } catch (SecurityException e) {
                canMockPosition = false;
            }
        }
        if (canMockPosition == false) {
            stopMockLocation();
        }
        return canMockPosition;
    }
```http://www.biyezuopin.vip
  接下來設置模擬經緯度數據:
```java
  // 模擬位置(addTestProvider成功的前提下)
  for (String providerStr : mockProviders) {
  Location mockLocation = new Location(providerStr);
  mockLocation.setLatitude(latitude);   // 維度(度)
  mockLocation.setLongitude(longitude);  // 經度(度)
  mockLocation.setAccuracy(0.1f);   // 精度(米)
  mockLocation.setTime(new Date().getTime());   // 本地時間
  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
    mockLocation.setElapsedRealtimeNanos(SystemClock.elapsedRealtimeNanos());
  }
    locationManager.setTestProviderLocation(providerStr, mockLocation);
  }
```http://www.biyezuopin.vip
  取消模擬定位方法:
```java
    /**
     * 取消位置模擬,以免啟用模擬數據後無法還原使用系統位置
     * 若模擬位置未開啟,則removeTestProvider將會拋出異常;
     * 若已addTestProvider後,關閉模擬位置,未removeTestProvider將導致系統GPS無數據更新;
     */
    public void stopMockLocation() {
        if (hasAddTestProvider) {
            for (String provider : mockProviders) {
                try {
                    locationManager.removeTestProvider(provider);
                } catch (Exception ex) {
                    // 此處不需要輸出日志,若未成功addTestProvider,則必然會出錯
                    // 這裡是對於非正常情況的預防措施
                }
            }
            hasAddTestProvider = false;
        }
    }

註冊位置服務,獲取系統位置

      // 註冊位置服務,獲取系統位置
        if (ActivityCompat.checkSelfPermission(context, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED && ActivityCompat.checkSelfPermission(context, Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
            //    ActivityCompat#requestPermissions
            // here to request the missing permissions, and then overriding
            //   public void onRequestPermissionsResult(int requestCode, String[] permissions,
            //                                          int[] grantResults)
            // to handle the case where the user grants the permission. See the documentation
            // for ActivityCompat#requestPermissions for more details.
            return;
        }
        mockLocationManager.locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, locationListener);

最後通過LocationListener.onLocationChanged()回調方法獲取GPS定位數據:

 private LocationListener locationListener = new LocationListener() {
        @Override
        public void onLocationChanged(final Location location) {
            setLocationData(location);
        }

        @Override
        public void onStatusChanged(String provider, int status, Bundle extras) {

        }

        @Override
        public void onProviderEnabled(String provider) {

        }

        @Override
        public void onProviderDisabled(String provider) {

        }
    };

    /**
     * 獲取到模擬定位信息,並顯示
     *
     * @param location 定位信息
     */
    private void setLocationData(Location location) {
        tvProvider.setText(location.getProvider());
        tvTime.setText(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date(location.getTime())));
        tvLatitude.setText(location.getLatitude() + " °");
        tvLongitude.setText(location.getLongitude() + " °");
    }

使用模擬定位需先開啟系統設置中的模擬位置:

Android 6.0 以下:【開發者選項 -> 允許模擬位置】

Android 6.0 及以上:【開發者選項 -> 選擇模擬位置信息應用】

到此這篇關於Android 模擬地圖定位功能的實現的文章就介紹到這瞭,更多相關Android 模擬地圖定位功能內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: