Android模擬定位

1 引言

模擬定位是為了解決軟件測試過程中需要對定位進行變更的需求問題。

1.1 步驟

1)設備需要先打開開發者模式,將其中的“允許模擬位置”打開(android6.0開始改為"選擇模擬位置信息應用")。
2)編寫或安裝相應的模擬定位的apk,實現模擬定位。

1.2 檢測方法

1)android6.0之前:使用Settings.Secure.ALLOW_MOCK_LOCATION判斷
2)android6.0開始:調用addTestProvider()方法,該方法在android.app.AppOpsManager#OPSTR_MOCK_LOCATION的值不為MODE_ALLOWED時會拋SecurityException異常。

2 模擬定位apk代碼

private LocationManager mLocManager;

mLocManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
mLocManager.addTestProvider(LocationManager.GPS_PROVIDER,
    "requiresNetwork" == "",
    "requiresSatellite" == "",
    "requiresCell" == "",
    "hasMonetaryCost" == "",
    "supportsAltitude" == "",
    "supportsSpeed" == "",
    "supportsBearing" == "",
    Criteria.NO_REQUIREMENT,
    Criteria.ACCURACY_COARSE);

// 創建新的Location對象,并設定必要的屬性值
Location newLocation = new Location(LocationManager.GPS_PROVIDER);
newLocation.setLatitude(39.820015);
newLocation.setLongitude(116.813752);
newLocation.setAccuracy(500);
newLocation.setTime(System.currentTimeMillis());
// 這里一定要設置nonasecond單位的值,否則是沒法持續收到監聽的
newLocation.setElapsedRealtimeNanos(SystemClock.elapsedRealtimeNanos());  

// 開啟測試Provider
mLocManager.setTestProviderEnabled(LocationManager.GPS_PROVIDER, true);

mLocManager.setTestProviderStatus(LocationManager.GPS_PROVIDER,
    LocationProvider.AVAILABLE,
    null,
    System.currentTimeMillis());

locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER,
                3000,
                10.0f,
                new MyLocationListener());

// 設置最新位置,一定要在requestLocationUpdate完成后進行,才能收到監聽
mLocManager.setTestProviderLocation(LocationManager.GPS_PROVIDER, newLocation);

代碼的大致流程為:addTestProvider-->創建新的Location對象并初始化-->setTestProviderEnabled-->setTestProviderStatus-->requestLocationUpdates-->setTestProviderLocation

3 源碼分析

源碼路徑:frameworks/base/location/java/android/location/LocationManager.java

3.1 addTestProvider

在LocationManager.java中該函數的主要流程為:

mService.addTestProvider(name, properties, mContext.getOpPackageName());

即調用mService的addTestProvider函數;
其中,mService為LocationManagerService
name為LocationManager.GPS_PROVIDER;
properties為傳入的參數組合而成的值;
getOpPackageName的結果用于給AppOpsManager標記該app;

接下來看LocationManagerService的addTestProvider():
源碼路徑: /frameworks/base/services/core/java/com/android/server/LocationManagerService.java

@Override
    public void addTestProvider(String name, ProviderProperties properties, String opPackageName) {
        //檢查權限,如果該app沒有
        if (!canCallerAccessMockLocation(opPackageName)) {
            return;
        }

        if (LocationManager.PASSIVE_PROVIDER.equals(name)) {
            throw new IllegalArgumentException("Cannot mock the passive location provider");
        }

        long identity = Binder.clearCallingIdentity();
        synchronized (mLock) {
            // remove the real provider if we are replacing GPS or network provider
            if (LocationManager.GPS_PROVIDER.equals(name)
                    || LocationManager.NETWORK_PROVIDER.equals(name)
                    || LocationManager.FUSED_PROVIDER.equals(name)) {
                LocationProviderInterface p = mProvidersByName.get(name);
                if (p != null) {
                    removeProviderLocked(p);
                }
            }
            addTestProviderLocked(name, properties);
            updateProvidersLocked();
        }
        Binder.restoreCallingIdentity(identity);
    }

\color{red}{其中的主要注釋為:當傳入的provider的名稱為GPS\_PROVIDER或其他兩個provider時,則會替換掉原本真正的provider。}由此,在后續獲取位置信息時,得到的就是模擬的定位信息了。

addTestProviderLocked

private void addTestProviderLocked(String name, ProviderProperties properties) {
        if (mProvidersByName.get(name) != null) {
            throw new IllegalArgumentException("Provider \"" + name + "\" already exists");
        }
        MockProvider provider = new MockProvider(name, this, properties);
        addProviderLocked(provider);
        mMockProviders.put(name, provider);
        //關鍵函數,此處將GPS_PROVIDER放入到mLastLocation,而mLastLocation是在獲取gps定位中存儲定位信息的對象
        mLastLocation.put(name, null);
        mLastLocationCoarseInterval.put(name, null);
    }

在其中創建一個MockProvider對象并加入到mMockProviders數組中。MockProvider的路徑為:
frameworks/base/services/core/java/com/android/server/location/MockProvider.java

3.2 setTestProviderStatus

public void setTestProviderStatus(String provider, int status, Bundle extras, long updateTime) {
        try {
            mService.setTestProviderStatus(provider, status, extras, updateTime,
                    mContext.getOpPackageName());
        } catch (RemoteException e) {
            throw e.rethrowFromSystemServer();
        }
    }

直接調用mService(即LocationManagerService)的setTestProviderStatus:

public void setTestProviderStatus(String provider, int status, Bundle extras, long updateTime,
            String opPackageName) {
        if (!canCallerAccessMockLocation(opPackageName)) {
            return;
        }

        synchronized (mLock) {
            MockProvider mockProvider = mMockProviders.get(provider);
            if (mockProvider == null) {
                throw new IllegalArgumentException("Provider \"" + provider + "\" unknown");
            }
            mockProvider.setStatus(status, extras, updateTime);
        }
    }

調用到mockProvider的setStatus函數,將狀態設為AVAILABLE

3.3 requestLocationUpdates

    @RequiresPermission(anyOf = {ACCESS_COARSE_LOCATION, ACCESS_FINE_LOCATION})
    public void requestLocationUpdates(String provider, long minTime, float minDistance,
            LocationListener listener, Looper looper) {
        checkProvider(provider);
        checkListener(listener);

        LocationRequest request = LocationRequest.createFromDeprecatedProvider(
                provider, minTime, minDistance, false);
        requestLocationUpdates(request, listener, looper, null);
    }

可以看到調用該函數是需要特定權限的,該函數的作用是設置定位更新的策略,此處傳參代表最短每3秒及每10m進行一次更新。

3.4 setTestProviderLocation

調用的是mService的setTestProviderLocation函數:

    @Override
    public void setTestProviderLocation(String provider, Location loc, String opPackageName) {
        if (!canCallerAccessMockLocation(opPackageName)) {
            return;
        }

        synchronized (mLock) {
            //從mockProviders里獲取GPS對應的mockProvider實例
            MockProvider mockProvider = mMockProviders.get(provider);
            if (mockProvider == null) {
                throw new IllegalArgumentException("Provider \"" + provider + "\" unknown");
            }

            // Ensure that the location is marked as being mock. There's some logic to do this in
            // handleLocationChanged(), but it fails if loc has the wrong provider (bug 33091107).
            Location mock = new Location(loc);
            mock.setIsFromMockProvider(true);

            if (!TextUtils.isEmpty(loc.getProvider()) && !provider.equals(loc.getProvider())) {
                // The location has an explicit provider that is different from the mock provider
                // name. The caller may be trying to fool us via bug 33091107.
                EventLog.writeEvent(0x534e4554, "33091107", Binder.getCallingUid(),
                        provider + "!=" + loc.getProvider());
            }

            // clear calling identity so INSTALL_LOCATION_PROVIDER permission is not required
            long identity = Binder.clearCallingIdentity();
            mockProvider.setLocation(mock);
            Binder.restoreCallingIdentity(identity);
        }
    }

mockProvider.setLocation(mock):

    public void setLocation(Location l) {
        mLocation.set(l);
        mHasLocation = true;
        if (mEnabled) {
            try {
                mLocationManager.reportLocation(mLocation, false);
            } catch (RemoteException e) {
                Log.e(TAG, "RemoteException calling reportLocation");
            }
        }
    }

調用到reportLocation(mLocation, false),即又回到LocationManagerService中,reportLocation的作用是向mLocationHandler發送一個MSG_LOCATION_CHANGED消息,并進行處理。
mLocationHandler接受到消息后調用handleLocationChanged函數再調用到handleLocationChangedLocked函數,在此處才會對定位信息設置為模擬的定位信息,由此,之后由GPS_PROVIDER獲取得到的定位信息都是模擬的定位信息。

?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,565評論 6 539
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,115評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,577評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,514評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,234評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,621評論 1 326
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,641評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,822評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,380評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,128評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,319評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,879評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,548評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,970評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,229評論 1 291
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,048評論 3 397
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,285評論 2 376