主頁 > 移動端開發 > 《第一行代碼:Android篇》學習筆記(十一)

《第一行代碼:Android篇》學習筆記(十一)

2022-05-12 08:32:53 移動端開發

本文和接下來的幾篇文章為閱讀郭霖先生所著《第一行代碼:Android(篇第2版)》的學習筆記,按照書中的內容順序進行記錄,書中的Demo本人全部都做過了,
每一章節本人都做了詳細的記錄,以下是我學習記錄(包含大量書中內容的整理和自己在學習中遇到的各種bug及解決方案),方便以后閱讀和查閱,最后,非常感激郭霖先生提供這么好的書籍,

第11章 Android特色開發——基于位置的服務

本章中,將要學習一些全新的Android技術,這些技術有別于傳統的PC或Web領域的應用技術,是只有在移動設備上才能實作的,

基于位置的服務(Location Based Service),由于移動設備相比于電腦可以隨身攜帶,我們通過地理定位的技術就可以隨時得知自己所在的位置,從而圍繞這一點開發出很多有意思的應用,

11.1 基于位置的服務簡介

基于位置的服務簡稱LBS,其實它本身并不是什么時髦的技術,主要的作業原理就是利用無線電通訊網路或GPS等定位方式來確定出移動設備所在的位置,而這種定位技術早在很多年前就已經出現了,

在過去移動設備的功能極其有限,即使定位到了設備所在的位置,也就僅僅只是定位到了而已,我們并不能在位置的基礎上進行一些其他的操作,而現在就大大不同了,有了Android系統作為載體,可以利用定位出的位置進行許多豐富多彩的操作,比如說天氣預報程式可以根據用戶所在的位置自動選擇城市,發微博的時候我們可以向朋友們曬一下自己在哪里,不認識路的時候隨時打開地圖就可以查詢路線,等等,

首先要清楚,基于位置的服務所圍繞的核心就是要先確定出用戶所在的位置,

通常有兩種技術方式可以實作,一種是通過GPS定位,一種是通過網路定位:

  • GPS定位的作業原理是基于手機內置的GPS硬體直接和衛星互動來獲取當前的經緯度資訊,這種定位方式精確度非常高,但缺點是只能在室外使用,室內基本無法接收到衛星的信號,
  • 網路定位的作業原理是根據手機當前網路附近的三個基站進行測速,以此計算出手機和每個基站之間的距離,再通過三角定位確定出一個大概的位置,這種定位方式精確度一般,但優點是在室內室外都可以使用,

Android對這兩種定位方式都提供了相應的API支持,但是由于一些特殊原因,Google的網路服務在中國不可訪問,從而導致網路定位方式的API失效,而GPS定位雖然不需要網路,但是必須要在室外才可以使用,因此你在室內開發的時候很有可能會遇到不管使用哪種定位方式都無法成功定位的情況,

基于以上原因,本書中不再講解Android原生定位API的用法了,而是使用一些國內第三方公司的SDK,目前國內在這一領域做得比較好的一個是百度,一個是高德,本章我們就來學習一下百度在LBS方面提供的豐富多彩的功能,

11.2 申請成為開發者

下面內容與書中記錄的不一樣了,書中的方式我注冊不了...現在記錄自己是如何申請成為百度開發者的流程:

要想在自己的應用程式里使用百度的LBS功能,你得擁有一個百度賬號才能進行申請,然后,登錄網住:http://lbsyun.baidu.com/apiconsole/key

image

填入真實資訊,進行郵箱驗證等,注冊的賬號:

image

結果在進行人臉識別認證的時候,我的始終無法識別,最后,申請了高德開發者,步驟如下:

1.網址:https://console.amap.com/dev/index

2.進行認證,支付寶、郵箱就可以了:

image

image

image

image

所以,以后Demo 書上是百度,我用的是高德了,

首先,創建新應用:

image

其次,新應用旁邊有一個添加按鈕,點擊添加:

image

這個發布版安全碼SHA1,開發模式(debug)和發布模式(release)下的 SHA1 值是不同的,發布 apk時 需要根據發布apk對應的keystore重新配置Key,獲取發布模式下的SHA1的方法(可點擊如何獲取,學習獲取步驟),通過Android Studio獲取測驗版SHA1:

1、打開Android Studio的Terminal工具,

2、輸入命令:keytool -v -list -keystore keystore檔案路徑,

3、輸入Keystore密碼,

image

image

其中,91:16:04:30:C0:8B:6E:53:92:47:57:E6:FB:10:EF:08:1B:73:E6:3E就是所需的SHA1指紋了,當然你的Android Studio中顯示的指紋和我的肯定是不一樣的,

另外需要注意,目前我們使用的是debug.keystore檔案所生成的指紋,這是Android自動生成的一個用于測驗的簽名檔案,而當你的應用程式發布時還需要創建一個正式的簽名檔案,如果要得到它的指紋,可以在cmd中輸入如下命令:

keytool -list -v -keystore <簽名檔案路徑>

然后輸入正確的密碼就可以了,創建簽名檔案的方法我們將在第15章中學習,

現在得到的這個SHA1指紋實際上是一個開發版的SHA1指紋,不過因為暫時我們還沒有一個發布版的SHA1指紋,因此這兩個值都填成一樣的就可以了,

上面這句話是個大坑!如果SHA1的發布版/測驗版值都填入一樣的,在最后很有可能出現:坐標一直為4.9E-324(這是導致定位坐標這樣的原因之一,我之后在測驗中,也改了不少,)

image

上面的代碼,與原書不一致,都是經過查詢網上資料改動過的,參考博客連接:

  1. 包名不一致(我的不是這個原因)https://www.cnblogs.com/hoyong/articles/11918642.html
  2. 室內用不了GPS....(我用的WIFI/移動4G)
  3. 網路不好,聯網失敗(我的網非常好,所以不是這個原因)
  4. Android的模擬器是不能定位的,所以要使用真機(我的不是這個原因,但是這個確實是真的,在我排查全部bug后,手機上測驗OK,到Android的模擬器上運行,為上圖:坐標一直為4.9E-324,不顯示定位方式)
  5. 權限錯誤/so庫加載錯誤(我的不是這個原因)https://www.jianshu.com/p/b394de3d989c

然而,在是上面3中的博客里,最后一條就是SHA1指紋發布版和測驗版不能一樣!!!我測驗了,對我來說是正確的解決辦法,

下面,記錄一下如何申請并找到SHA1指紋發布版步驟:

  1. 找到apk簽名路徑,可通過AS中build-generate signed bundle/apk -apk 進入如圖界面:

image

image

  1. 檢查一下該目錄(E:\androidKey)中是否存在zhouzhou.jks,沒有的話,查看檔案是否被隱藏了:

image

其中key store path 為簽名路徑,

  1. 操作和獲取開發版SHA1 基本一致,只需將debug.keystore換成簽名路徑即可,如圖:

image

  1. 最后,將在開發平臺更換發布版SHA1重新提交,注:專案最好clean一下,

下面開始,繼續記錄書上的內容,

最后還剩下一個包名選項,雖然目前我們的應用程式還不存在,但可以先將包名預定下來,比如就叫com.zhouzhou. lbstest,這樣所有的內容就都填寫完整了,如圖為百度地圖開放平臺創建的應用資訊:

(注:第二天百度的人臉識別認證我整成功了,為了和書上保持一致,我接下來選擇用百度地圖開放平臺做Demo,事實上,高德平臺步驟也差不多,以后有機會會嘗試的)

image

接下來點擊提交,應用就應該創建成功了,如圖:

image

其中,i6VD2fHKM3msMfZtIOXAhFSzDiYGFIwL(訪問應用(AK))就是申請到的API Key,有了它就可以進行后續的LBS開發作業了,那么我們馬上開始吧,

11.3 使用百度定位

新建一個LBSTest專案,包名應該就會自動被命名為com.zhouzhou.lbstest,另外需要注意,本章中所寫的代碼建議都在手機上運行,雖然模擬器中也提供了模擬地理位置的功能,但在手機上可以得到真實的位置資料,你的感受會更加深刻,

11.3.1 準備LBS SDK

在開始編碼之前,我們還需要先將百度LBS開放平臺的SDK準備好,目前百度地圖Android地圖SDK境內服務對非商業目的使用的開發者不收取任何費用,開發者可自行下載:http://lbsyun.baidu.com/index.php?title=androidsdk/sdkandev-download放心使用,

image

本章中,我們會用到基礎地圖和定位功能這兩個SDK,將它們勾選上,然后點擊“開發包”下載按鈕即可,如圖:

image

下載完成后對該壓縮包解壓,其中會有一個libs目錄,這里面的內容就是我們所需要的一切了,如圖:

image

libs目錄下的內容又分為兩部分,BaiduLBS_Android.jar這個檔案是Java層要使用到的,其他子目錄下的so檔案是Native層要用到的,so檔案是用C/C++語言進行撰寫,然后再用NDK編譯出來的,當然這里我們并不需要去撰寫C/C++的代碼,因為百度都已經做好了封裝,但是我們需要將libs目錄下的每一個檔案都放置到正確的位置,

首先,觀察一下當前的專案結構,會發現app模塊下面有一個libs目錄,這里就是用來存放所有的Jar包的,我們將BaiduLBS_Android.jar復制到這里,如圖:

image

接下來展開src/main目錄,右擊該目錄→New→Directory,再創建一個名為jniLibs的目錄,這里就是專門用來存放so檔案的,然后把壓縮包里的其他所有目錄直接復制到這里,如圖:

image

另外,雖然所有新創建的專案中,app/build.gradle檔案都會默認配置以下這段宣告(不整這個也行,上面BaiduLBS_Android.jar復制之后,直接Add As Library...即可,dependencies中會被自動添加一行implementation files('libs\\BaiduLBS_Android.jar'),注意,Add As Library...和下面的implementation fileTree(include: ['*.jar'], dir: 'libs')方式要二選一,不要都寫上,會報沖突!!!):

dependencies {
    implementation fileTree(include: ['*.jar'], dir: 'libs')
}

這表示會將libs目錄下所有以.jar結尾的檔案添加到當前專案的參考中,(但是由于我們是直接將Jar包復制到libs目錄下的,并沒有修改gradle檔案,因此不會彈出我們平時熟悉的Sync Now提示,這個時候必須手動點擊一下AndroidStudio頂部工具列中的Sync按鈕,不然專案將無法參考到Jar包中提供的任何介面,)

點擊Sync按鈕之后,libs目錄下的jar檔案就會多出一個向右的箭頭,這就表示專案已經能參考到這些Jar包了,如圖:

image

這樣我們就把LBS的SDK都準備好了,接下來開始編碼吧,

11.3.2 確定自己位置的經緯度

首先修改activity_main.xml中的代碼,如下所示:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/position_text_view"
        android:layout_
        android:layout_height="wrap_content"/>
</LinearLayout>

然后,修改AndroidManifest.xml檔案中的代碼,如下所示:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    package="com.zhouzhou.lbstest">
    <!-- 這個權限用于進行網路定位-->
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
    <!-- 這個權限用于訪問GPS定位-->
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
    <!-- 用于訪問wifi網路資訊,wifi資訊會用于進行網路定位-->
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
    <!-- 獲取運營商資訊,用于支持提供運營商資訊相關的介面-->
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
    <!-- 這個權限用于獲取wifi的獲取權限,wifi資訊會用來進行網路定位-->
    <uses-permission android:name="android.permission.CHANGE_WIFI_STATE"/>
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <!-- 寫入擴展存盤,向擴展卡寫入資料,用于寫入離線定位資料-->
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    <!-- 訪問網路,網路定位需要上網-->
    <uses-permission android:name="android.permission.INTERNET"/>
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"
        tools:ignore="ProtectedPermissions" />
    <uses-permission android:name="android.permission.WAKE_LOCK" />
    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/Theme.LBSTest">

        <!-- meta-data中android:value=https://www.cnblogs.com/1693977889zz/p/在11.2節申請到的API Key -->
        <meta-data
            android:name="com.baidu.lbsapi.API_KEY"
            android:value="https://www.cnblogs.com/1693977889zz/p/LGvkANF1s4zZK1MkK0XL6fYiL2TkV4L4" />
        <activity
            android:name=".MainActivity"
            android:exported="true">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <service
            android:name="com.baidu.location.f"
            android:enabled="true"
            android:process=":remote">
        </service>
    </application>

</manifest>

AndroidManifest.xml檔案改動比較多,可以看到,這里首先添加了很多行權限宣告,每一個權限都是百度LBS SDK內部要用到的,然后,在<application>標簽的內部添加了一個<meta-data>標簽,這個標簽的android:name部分是固定的,必須填com.baidu. lbsapi.API_KEY,android:value部分則應該填入我們在11.2節申請到的API Key,最后,還需要再注冊一個LBS SDK中的服務,不用對這個服務的名字感到疑惑,因為百度LBS SDK中的代碼都是混淆過的,

接下來修改MainActivity中的代碼,如下所示:

package com.zhouzhou.lbstest;

import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;

import android.Manifest;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.widget.TextView;
import android.widget.Toast;

import com.baidu.location.BDLocation;
import com.baidu.location.BDLocationListener;
import com.baidu.location.LocationClient;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity {
    public LocationClient mLocationClient;
    private TextView positionText;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //沒有它,會報錯:Please recheck the setAgreePrivacy interface
        mLocationClient.setAgreePrivacy(true);
        try {
            //首先創建了一個LocationClient的實體,LocationClient的構建函式接收一個Context引數,
            //這里呼叫getApplicationContext()方法來獲取一個全域的Context引數并傳入,
            mLocationClient = new LocationClient(getApplicationContext());
        } catch (Exception e) {
                e.printStackTrace();
            }
            //然后呼叫LocationClient的registerLocationListener()方法來注冊一個定位監聽器,當獲取到位置資訊的時候,就會回呼這個定位監聽器,
            mLocationClient.registerLocationListener(new MyLocationListener());
            setContentView(R.layout.activity_main);
            positionText = (TextView) findViewById(R.id.position_text_view);
            //怎樣才能在運行時一次性申請3個權限呢?
            //創建一個空的List集合,然后依次判斷這3個權限有沒有被授權,
            //如果沒被授權就添加到List集合中,最后將List轉換成陣列,
            //再呼叫ActivityCompat.requestPermissions()方法一次性申請,
            List<String> permissionList = new ArrayList<>();
            if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
                permissionList.add(Manifest.permission.ACCESS_FINE_LOCATION);
            }
            if (ContextCompat.checkSelfPermission(MainActivity.this,Manifest.permission.READ_PHONE_STATE) != PackageManager.PERMISSION_GRANTED) {
                permissionList.add(Manifest.permission.READ_PHONE_STATE);
            }
            if (ContextCompat.checkSelfPermission(MainActivity.this,Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
                permissionList.add(Manifest.permission.WRITE_EXTERNAL_STORAGE);
            }
            if (! permissionList.isEmpty()) {
                String [] permissions = permissionList.toArray(new String[permissionList.size()]);
                ActivityCompat.requestPermissions(MainActivity.this,permissions,1);
            } else {
                requestLocation();
            }
    }
    private void requestLocation() {
        mLocationClient.start();
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        switch (requestCode) {
            case 1:
                if (grantResults.length > 0) {
                    for (int result : grantResults) {
                        if (result != PackageManager.PERMISSION_GRANTED) {
                            Toast.makeText(this,"必須同意所有權限才能使用本程式",Toast.LENGTH_SHORT).show();
                            finish();
                            return;
                        }
                    }
                    requestLocation();
                } else {
                    Toast.makeText(this,"發生未知錯誤",Toast.LENGTH_SHORT).show();
                    finish();
                }
                break;
            default:
        }
    }

    public class MyLocationListener implements BDLocationListener {
        /**
         * MyLocationListener的onReceiveLocation()方法中:
         * 通過BDLocation的getLatitude()方法獲取當前位置的緯度,
         * 通過getLongitude()方法獲取當前位置的經度,
         * 通過getLocType()方法獲取當前的定位方式,最終將結果組裝成一個字串,顯示到TextView上面,
         */
        @Override
        public void onReceiveLocation(BDLocation bdLocation) {
            runOnUiThread(new Runnable() {
                @Override
                public void run() {
                    StringBuilder currentPosition = new StringBuilder();
                    currentPosition.append("緯度:").append(bdLocation.getLatitude()).append("\n");
                    currentPosition.append("經度:").append(bdLocation.getLongitude()).append("\n");
                    currentPosition.append("定位方式:");
                    if (bdLocation.getLocType() == BDLocation.TypeGpsLocation) {
                        currentPosition.append("GPS");
                    } else if (bdLocation.getLocType() == BDLocation.TypeNetWorkLocation) {
                        currentPosition.append("網路");
                    }
                    positionText.setText(currentPosition);
                }
            });
        }
    }
}

注意:上面代碼中多了一行:

mLocationClient.setAgreePrivacy(true);

如果沒有這一行,報錯:...Please recheck the setAgreePrivacy interface,忘記截圖了,

  1. 問題描述: 根據官方檔案配置完后,地圖無法顯示,也不顯示網格圖層,會報location呼叫隱私政策介面例外,
  2. 解決辦法: 除了在Application.java中正常初始化context資訊之外,在呼叫地圖的activity中的onCreat()方法中,添加一句LocationClient.setAgreePrivacy(true);
  3. 參考博客:解決百度地圖sdk v7.5.0版本,地圖不能正常顯示問題(隱私政策)https://blog.csdn.net/qq_43962511/article/details/123840858

運行時權限的用法,由于我們在AndroidManifest.xml中宣告了很多權限,參考一下7.2.1小節中的危險權限表格可以發現,其中ACCESS_COARSE_LOCATION、ACCESS_FINE_LOCATION、READ_PHONE_STATE、WRITE_EXTERNAL_STORAGE這4個權限是需要進行運行時權限處理的,不過由于ACCESS_COARSE_LOCATION和ACCESS_FINE_LOCATION都屬于同一個權限組,因此兩者只要申請其一就可以了,

那么怎樣才能在運行時一次性申請3個權限呢?

這里我們使用了一種新的用法,首先創建一個空的List集合,然后依次判斷這3個權限有沒有被授權,如果沒被授權就添加到List集合中,最后將List轉換成陣列,再呼叫ActivityCompat. requestPermissions()方法一次性申請,

除此之外,onRequestPermissionsResult()方法中對權限申請結果的邏輯處理也和之前有所不同,這次我們通過一個回圈將申請的每個權限都進行了判斷,如果有任何一個權限被拒絕,那么就直接呼叫finish()方法關閉當前程式,只有當所有權限都被用戶同意了,才會呼叫requestLocation()方法開始地理位置定位,

requestLocation()方法中的代碼比較簡單,只是呼叫了一下LocationClient的start()方法就能開始定位了,定位的結果會回呼到我們前面注冊的監聽器當中,也就是MyLocationListener,

現在運行一下程式了,毫無疑問,打開程式首先就會彈出運行時權限的申請對話框,注意,與書中不同,我的Demo手機上是一項一項選擇的是否同意(3次授權):

image

然后就會立刻開始定位了,結果如圖:

image

可以看到,設備當前的經緯度資訊已經成功定位出來了,

不過,在默認情況下,呼叫LocationClient的start()方法只會定位一次,如果我們正在快速移動中,怎樣才能實時更新當前的位置呢?

為此,百度LBS SDK提供了一系列的設定方法,來允許我們更改默認的行為,修改MainActivity中的代碼,如下所示:

public class MainActivity extends AppCompatActivity {
    ...
    private void requestLocation() {
        initLocation();
        mLocationClient.start();
    }
    private void initLocation() {
        LocationClientOption option = new LocationClientOption();
        // 表示每5秒鐘會更新一下當前的位置
        option.setScanSpan(5000);
        mLocationClient.setLocOption(option);
    }
    @Override
    protected void onDestroy() {
        super.onDestroy();
        mLocationClient.stop();
    }
    ...
}

這里增加了一個initLocation()方法,在initLocation()方法中我們創建了一個LocationClientOption物件,然后呼叫它的setScanSpan()方法來設定更新的間隔,

最后要記得,在活動被銷毀的時候一定要呼叫LocationClient的stop()方法來停止定位,不然程式會持續在后臺不停地進行定位,從而嚴重消耗手機的電量,

重新運行程式,拿著手機隨處移動,界面上的經緯度資訊也會跟著一起變化,(測驗,有效!!!)

11.3.3 選擇定位模式

Android中主要有兩種:一種是通過GPS定位,一種是通過網路定位,

上一小節中的例子中,一直是使用的網路定位,那么如何才能切換到精確度更高的GPS定位呢?

本小節我們就來學習一下,首先,GPS定位功能必須要由用戶主動去啟用才行,不然任何應用程式都無法使用GPS獲取到手機當前的位置資訊,進入手機的設定→位置資訊(我的手機沒有位置資訊選項,下圖是我Android的模擬器上的截取的),如圖:

image

可以通過頂部的開關來控制定位功能是開啟還是關閉,另外,點擊“模式”可以選擇具體的定位模式,我的手機與書上的不一樣,我下拉手機選單欄,看到GPS,手指長按GPS跳到GPS設定界面,如下圖所示:

image

其中,

  • 高精確度模式,表示允許使用GPS、無線網路、藍牙或移動網路來進行定位;
  • 節電模式,表示僅允許使用無線網路、藍牙或移動網路來進行定位;
  • 僅限設備模式,表示僅允許使用GPS來進行定位,

也就是說,如果我們想要使用GPS定位功能,這里必須要選擇高精確度模式,或者僅限設備模式,

當然,你并不需要擔心一旦啟用GPS定位功能后,手機的電量就會直線下滑,這只是表明你已經同意讓應用程式來對你的手機進行GPS定位了,但只有當定位操作真正開始的時候,才會影響到手機的電量,

開啟了GPS定位功能之后,再回來看一下代碼,我們可以在initLocation()方法中對百度LBS SDK的定位模式進行指定,一共有3種模式可選:Hight_Accuracy、Battery_Saving和Device_Sensors,

  • Hight_Accuracy 表示高精確度模式,會在GPS信號正常的情況下優先使用GPS定位,在無法接收GPS信號的時候使用網路定位,
  • Battery_Saving 表示節電模式,只會使用網路進行定位,
  • Device_Sensors 表示傳感器模式,只會使用GPS進行定位,

Hight_Accuracy是默認的模式,也就是說,我們即使不修改任何代碼,只要拿著手機走到室外去,讓手機可以接收到GPS信號,就會自動切換到GPS定位模式了,

當然我們也可以強制指定只使用GPS進行定位,修改MainActivity中的代碼,如下所示:

public class MainActivity extends AppCompatActivity {
   ...
    private void initLocation() {
        LocationClientOption option = new LocationClientOption();
        // 表示每5秒鐘會更新一下當前的位置
        option.setScanSpan(5000);
       // 呼叫了setLocationMode()方法來將定位模式指定成傳感器模式,也就是說只能使用GPS進行定位
        option.setLocationMode(LocationClientOption.LocationMode.Device_Sensors);
        mLocationClient.setLocOption(option);
    }
    @Override
    protected void onDestroy() {
        super.onDestroy();
        mLocationClient.stop();
    }
	...
}

這里呼叫了setLocationMode()方法來將定位模式指定成傳感器模式,也就是說只能使用GPS進行定位,重新運行一下程式,然后拿著你的手機走到室外去,切記:如果想要使用GPS定位功能,手機上的GPS模式,必須要選擇高精確度模式,或者僅限設備模式 ,結果如圖:

image

11.3.4 看得懂的位置資訊

雖然成功獲取到了設備當前位置的經緯度資訊,但這種經緯度的值一般人是根本看不懂,

為了能夠更加直觀地閱讀,我們還需要學習一下如何獲取看得懂的位置資訊,百度LBS SDK在這方面提供了非常好的支持,我們只需要進行一些簡單的介面呼叫就能得到當前位置各種豐富的地址資訊,

修改MainActivity中的代碼,如下所示:

public class MainActivity extends AppCompatActivity {
    ...
    private void initLocation() {
        LocationClientOption option = new LocationClientOption();
        // 表示每5秒鐘會更新一下當前的位置
        option.setScanSpan(5000);
        // 呼叫了LocationClientOption的setIsNeedAddress()方法,并傳入true,表示需要獲取當前位置詳細的地址資訊,
        option.setIsNeedAddress(true);
        // 呼叫了setLocationMode()方法來將定位模式指定成傳感器模式,也就是說只能使用GPS進行定位
        option.setLocationMode(LocationClientOption.LocationMode.Device_Sensors);
        mLocationClient.setLocOption(option);
    }
    ...
    public class MyLocationListener implements BDLocationListener {
        @Override
        public void onReceiveLocation(BDLocation bdLocation) {
            runOnUiThread(new Runnable() {
                @Override
                public void run() {
                    StringBuilder currentPosition = new StringBuilder();
                    currentPosition.append("緯度:").append(bdLocation.getLatitude()).append("\n");
                    currentPosition.append("經度:").append(bdLocation.getLongitude()).append("\n");
                    currentPosition.append("國家:").append(bdLocation.getCountry()).append("\n");
                    currentPosition.append("省:").append(bdLocation.getProvince()).append("\n");
                    currentPosition.append("市:").append(bdLocation.getCity()).append("\n");
                    currentPosition.append("區:").append(bdLocation.getDistrict()).append("\n");
                    currentPosition.append("街道:").append(bdLocation.getStreet()).append("\n");
                    currentPosition.append("定位方式:");
                    if (bdLocation.getLocType() == BDLocation.TypeGpsLocation) {
                        currentPosition.append("GPS");
                    } else if (bdLocation.getLocType() == BDLocation.TypeNetWorkLocation) {
                        currentPosition.append("網路");
                    }
                    positionText.setText(currentPosition);
                }
            });
        }
    }
}

在MyLocationListener的onReceiveLocation()方法就可以獲取到各種豐富的地址資訊了,呼叫getCountry()方法可以得到當前所在國家,呼叫getProvince()方法可以得到當前所在省份,以此類推,

需要注意,由于獲取地址資訊一定需要用到網路,因此即使我們將定位模式指定成了Device_Sensors,也會自動開啟網路定位功能,現在重新運行一下程式,結果如圖:

image

可以看到,手機當前位置的地址資訊已經成功顯示出來了,如果你帶著手機移動了較遠的距離,界面上顯示的位置也會跟著一起變化的,

11.4 使用百度地圖

手機地圖能夠隨時隨地進行查看,并且輕松構建出行路線,使用起來明顯更加地方便,應用程式里也是可以加入地圖功能的,比如優步中使用的就是百度地圖,本節我們就來學習一下這方面的知識,

11.4.1 讓地圖顯示出來

讓地圖顯示出來,這個例子我決定我像書中介紹的再LBSTest應用中做了,想再自己做一遍使用百度地圖的流程,所以又新建了一個LBSTest2應用,用這個來記錄一下,

  • 第一步:Android Studio 中:

    • 新建LBSTest2應用

    • 下載SDK本地依賴

    • 將開發包拷貝至工程

      • 添加jar檔案(BaiduLBS_Android.jar),Add As Library...

      • 添加so檔案(在src/main/目錄下新建jniLibs目錄(如果您的專案中已經包含該目錄不用重復創建),在下載的開發包中拷貝專案中需要的CPU架構對應的so檔案檔案夾到jniLibs目錄)

        注意:Jar檔案和so檔案的版本號必須一致,并且保證Jar檔案與so檔案是同一版本包取出的,

image

(注:第一步主要參考官方檔案地址的地址:https://lbsyun.baidu.com/index.php?title=androidsdk/guide/create-project/androidstudio

  • 第二步:在百度地圖開放平臺,創建應用,獲得訪問應用(AK):

    (注:第二步一下,結合的官方開發檔案地址:https://lbsyun.baidu.com/index.php?title=androidsdk/guide/create-map/showmap )

image

  • 第三步:配置AndroidManifest.xml檔案
<application>  
    <meta-data  
        android:name="com.baidu.lbsapi.API_KEY"  
        android:value="https://www.cnblogs.com/1693977889zz/p/開發者 key" />  
</application>
  • 第四步: 在<application/>外部添加如下權限宣告:
<!-- 訪問網路,進行地圖相關業務資料請求,包括地圖資料,路線規劃,POI檢索等 -->
<uses-permission android:name="android.permission.INTERNET" />
<!-- 獲取網路狀態,根據網路狀態切換進行資料請求網路轉換 -->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

<!-- 讀取外置存盤,如果開發者使用了so動態加載功能并且把so檔案放在了外置存盤區域,則需要申請該權限,否則不需要 -->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<!-- 寫外置存盤,如果開發者使用了離線地圖,并且資料寫在外置存盤區域,則需要申請該權限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

注:自Android6.0起部分權限的使用需要開發者在代碼中動態申請,

參考博客:解決百度地圖sdk v7.5.0版本,地圖不能正常顯示問題,https://blog.csdn.net/qq_43962511/article/details/123840858

// 是否同意隱私政策,默認為false
SDKInitializer.setAgreePrivacy(getApplicationContext(), true);
  • 第五步: 在布局檔案中添加地圖容器

MapView是View的一個子類,用于在Android View中放置地圖,MapView的使用方法與Android提供的其他View一樣,

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_
    android:layout_height="match_parent">
    <com.baidu.mapapi.map.MapView
        android:id="@+id/bmapView"
        android:layout_
        android:layout_height="match_parent"
        android:clickable="true" />
</LinearLayout>
  • 第六步:地圖初始化

注意:在SDK各功能組件使用之前都需要呼叫“SDKInitializer.initialize(getApplicationContext())”,因此建議在應用創建時初始化SDK參考的Context為全域變數,

新建一個自定義的Application——DemoApplication,在其onCreate方法中完成SDK的初始化,示例代碼如下:

(其中,“是否同意隱私政策”是之前沒有添加,出來bug后,上網查到的,try/catch 是網上查到的)

package com.zhouzhou.lbstest2;

import android.app.Application;

import com.baidu.mapapi.CoordType;
import com.baidu.mapapi.SDKInitializer;
import com.baidu.mapapi.common.BaiduMapSDKException;

public class DemoApplication extends Application {
    @Override
    public void onCreate() {
        super.onCreate();
        // 是否同意隱私政策,默認為false
        SDKInitializer.setAgreePrivacy(getApplicationContext(), true);
        try {
            //在使用SDK各組件之前初始化context資訊,傳入ApplicationContext
            SDKInitializer.initialize(getApplicationContext());
            //自4.3.0起,百度地圖SDK所有介面均支持百度坐標和國測局坐標,用此方法設定您使用的坐標型別.
            //包括BD09LL和GCJ02兩種坐標,默認是BD09LL坐標,
            SDKInitializer.setCoordType(CoordType.BD09LL);

        } catch (BaiduMapSDKException e) {
            e.printStackTrace();
        }
    }
}
  • 第七步:在AndroidManifest.xml檔案中宣告該Application
<application
       android:name=".DemoApplication"

image

  • 第八步:創建地圖Activity,管理MapView生命周期

注意:在專案中使用地圖的時候要特別注意合理地管理地圖生命周期,這非常重要,

以下示例代碼簡述對地圖生命周期的管理:

package com.zhouzhou.lbstest2;

import androidx.appcompat.app.AppCompatActivity;

import android.os.Bundle;
import com.baidu.mapapi.map.MapView;

public class MainActivity extends AppCompatActivity {
    private MapView mMapView = null;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //獲取地圖控制元件參考
        mMapView = (MapView) findViewById(R.id.bmapView);
    }
    @Override
    protected void onResume() {
        super.onResume();
        //在activity執行onResume時執行mMapView. onResume (),實作地圖生命周期管理
        mMapView.onResume();
    }
    @Override
    protected void onPause() {
        super.onPause();
        //在activity執行onPause時執行mMapView. onPause (),實作地圖生命周期管理
        mMapView.onPause();
    }
    @Override
    protected void onDestroy() {
        super.onDestroy();
        //在activity執行onDestroy時執行mMapView.onDestroy(),實作地圖生命周期管理
        mMapView.onDestroy();
    }
}

完成以上作業即可在您的應用中顯示地圖:

image

11.4.2 移動到我的位置

這是一張默認的地圖,顯示的是北京市中心的位置,而你可能希望看到更加精細的地圖資訊,比如說自己所在位置的周邊環境,顯然,通過縮放和移動的方式來慢慢找到自己的位置是一種很愚蠢的做法,那么本小節我們就來學習一下,如何才能在地圖中快速移動到自己的位置,

百度LBS SDK的API中提供了一個BaiduMap類,它是地圖的總控制器,呼叫MapView的getMap()方法就能獲取到BaiduMap的實體,如下所示:

BaiduMap baiduMap = mapView.getMap();

有了BaiduMap后,我們就能對地圖進行各種各樣的操作了,比如設定地圖的縮放級別以及將地圖移動到某一個經緯度上,

百度地圖將縮放級別的取值范圍限定在3到19之間,其中小數點位的值也是可以取的,值越大,地圖顯示的資訊就越精細,比如我們想要將縮放級別設定成12.5,就可以這樣寫:

MapStatusUpdate update = MapStatusUpdateFactory.zoomTo(12.5f);
baiduMap.animateMapStatus(update);

其中MapStatusUpdateFactory的zoomTo()方法接收一個float型的引數,就是用于設定縮放級別的,這里我們傳入12.5f,zoomTo()方法回傳一個MapStatusUpdate物件,我們把這個物件傳入BaiduMap的animateMapStatus()方法當中即可完成縮放功能,

那么怎樣才能讓地圖移動到某一個經緯度上呢?

這就需要借助LatLng類,其實LatLng并沒有什么太多的用法,主要就是用于存放經緯度值的,它的構造方法接收兩個引數,第一個引數是緯度值,第二個引數是經度值,之后呼叫MapStatusUpdateFactory的newLatLng()方法將LatLng物件傳入,newLatLng()方法回傳的也是一個MapStatusUpdate物件,我們再把這個物件傳入BaiduMap的animateMapStatus()方法當中,就可以將地圖移動到指定的經緯度上了,寫法如下:

LatLng ll = new LatLng(39.915,116.404);
MapStatusUpdate update = MapStatusUpdateFactory.newLatLng(ll);
baiduMap.animateMapStatus(update);

上述代碼就實作了將地圖移動到北緯39.915度、東經116.404度這個位置的功能,

了解了這些知識之后,接下來再去實作將地圖快速移動到自己位置的功能就變得非常簡單了,首先我們可以利用在11.3節中所學的定位技術來獲得自己當前位置的經緯度,之后再按照上述的方法來將地圖移動到指定的位置就可以了,

那么下面我們就來繼續完善LBSTest這個專案,加入“移動到我的位置”這個功能,修改MainActivity中的代碼,如下所示:

package com.zhouzhou.lbstest2;

import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;

import android.Manifest;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.widget.Toast;

import com.baidu.location.BDLocation;
import com.baidu.location.BDLocationListener;
import com.baidu.location.LocationClient;
import com.baidu.mapapi.map.BaiduMap;
import com.baidu.mapapi.map.MapStatusUpdate;
import com.baidu.mapapi.map.MapStatusUpdateFactory;
import com.baidu.mapapi.map.MapView;
import com.baidu.mapapi.model.LatLng;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity {
    public LocationClient mLocationClient;
    private MapView mMapView = null;
    private BaiduMap baiduMap;
    private boolean isFirstLocate = true;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //沒有它,會報錯:Please recheck the setAgreePrivacy interface
        mLocationClient.setAgreePrivacy(true);
        try {
            mLocationClient = new LocationClient(getApplicationContext());
        } catch (Exception e) {
            e.printStackTrace();
        }
        mLocationClient.registerLocationListener(new MyLocationListener());
        setContentView(R.layout.activity_main);
        //獲取地圖控制元件參考
        mMapView = (MapView) findViewById(R.id.bmapView);
        baiduMap = mMapView.getMap();
        List<String> permissionList = new ArrayList<>();
        if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
            permissionList.add(Manifest.permission.ACCESS_FINE_LOCATION);
        }
        if (ContextCompat.checkSelfPermission(MainActivity.this,Manifest.permission.READ_PHONE_STATE) != PackageManager.PERMISSION_GRANTED) {
            permissionList.add(Manifest.permission.READ_PHONE_STATE);
        }
        if (ContextCompat.checkSelfPermission(MainActivity.this,Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
            permissionList.add(Manifest.permission.WRITE_EXTERNAL_STORAGE);
        }
        if (! permissionList.isEmpty()) {
            String [] permissions = permissionList.toArray(new String[permissionList.size()]);
            ActivityCompat.requestPermissions(MainActivity.this,permissions,1);
        } else {
            requestLocation();
        }
    }
    private void requestLocation() {
        mLocationClient.start();
    }
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        switch (requestCode) {
            case 1:
                if (grantResults.length > 0) {
                    for (int result : grantResults) {
                        if (result != PackageManager.PERMISSION_GRANTED) {
                            Toast.makeText(this,"必須同意所有權限才能使用本程式",Toast.LENGTH_SHORT).show();
                            finish();
                            return;
                        }
                    }
                    requestLocation();
                } else {
                    Toast.makeText(this,"發生未知錯誤",Toast.LENGTH_SHORT).show();
                    finish();
                }
                break;
            default:
        }
    }
    private void navigateTo(BDLocation location) {
        if (isFirstLocate) {
            LatLng ll = new LatLng(location.getLatitude(),location.getLongitude());
            MapStatusUpdate update = MapStatusUpdateFactory.newLatLng(ll);
            baiduMap.animateMapStatus(update);
            update = MapStatusUpdateFactory.zoomTo(16f);
            baiduMap.animateMapStatus(update);
            isFirstLocate = false;
        }
    }
    public class MyLocationListener implements BDLocationListener {

        @Override
        public void onReceiveLocation(BDLocation bdLocation) {
            if (bdLocation.getLocType() == BDLocation.TypeGpsLocation || bdLocation.getLocType() == BDLocation.TypeNetWorkLocation) {
                navigateTo(bdLocation);
            }
        }
    }
    @Override
    protected void onResume() {
        super.onResume();
        //在activity執行onResume時執行mMapView. onResume (),實作地圖生命周期管理
        mMapView.onResume();
    }
    @Override
    protected void onPause() {
        super.onPause();
        //在activity執行onPause時執行mMapView. onPause (),實作地圖生命周期管理
        mMapView.onPause();
    }
    @Override
    protected void onDestroy() {
        super.onDestroy();
        mLocationClient.stop();
        //在activity執行onDestroy時執行mMapView.onDestroy(),實作地圖生命周期管理
        mMapView.onDestroy();
    }
}

主要是加入了一個navigateTo()方法,這個方法,先是將BDLocation物件中的地理位置資訊取出并封裝到LatLng物件中,然后呼叫MapStatusUpdateFactory的newLatLng()方法并將LatLng物件傳入,接著將回傳的MapStatus-Update物件作為引數傳入到BaiduMap的animateMapStatus()方法當中,和上面介紹的用法是一模一樣的,

并且,這里為了讓地圖資訊可以顯示得更加豐富一些,我們將縮放級別設定成了16,需要注意,上述代碼當中我們使用了一個isFirstLocate變數,這個變數的作用是為了防止多次呼叫animateMapStatus()方法,因為將地圖移動到我們當前的位置只需要在程式第一次定位的時候呼叫一次就可以了,

另外,當定位到設備當前位置的時候,我們在onReceiveLocation()方法中直接把BDLocation物件傳給navigateTo()方法,這樣就能夠讓地圖移動到設備所在的位置了,

注意:出現一些因為沒有添加權限而產生的bug,所以,在AndroidManifest.xml中又添加了幾項權限,整體如下:

<!-- 訪問網路,進行地圖相關業務資料請求,包括地圖資料,路線規劃,POI檢索等 -->
<uses-permission android:name="android.permission.INTERNET" />
<!-- 獲取網路狀態,根據網路狀態切換進行資料請求網路轉換 -->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

<!-- 讀取外置存盤,如果開發者使用了so動態加載功能并且把so檔案放在了外置存盤區域,則需要申請該權限,否則不需要 -->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<!-- 寫外置存盤,如果開發者使用了離線地圖,并且資料寫在外置存盤區域,則需要申請該權限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<!-- 這個權限用于進行網路定位 -->
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<!-- 這個權限用于訪問GPS定位 -->
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<!-- WIFI -->
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE"/>
<!-- PHONE -->
<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
<uses-permission android:name="android.permission.CALL_PHONE"/>

現在重新運行一下程式,結果如圖:

image

11.4.3 讓“我”顯示在地圖上

已經可以讓地圖顯示我們周邊的環境了,通常情況下手機地圖上應該都會有一個小游標,用于顯示設備當前所在的位置,并且如果設備正在移動的話,那么這個游標也會跟著一起移動,那么我們現在就繼續對現有代碼進行擴展,讓“我”能夠顯示在地圖上,

百度LBS SDK當中提供了一個MyLocationData.Builder類,這個類是用來封裝設備當前所在位置的,我們只需將經緯度資訊傳入到這個類的相應方法當中就可以了,如下所示:

MyLocationData.Builder locationBuilder = new MyLocationData.Builder();
locationBuilder.latitude(39.915);
locationBuilder.longitude(116.404);

MyLocationData.Builder類還提供了一個build()方法,當我們把要封裝的資訊都設定完成之后,只需要呼叫它的build()方法,就會生成一個MyLocationData的實體,然后再將這個實體傳入到BaiduMap的setMyLocationData()方法當中,就可以讓設備當前的位置顯示在地圖上了,寫法如下:

MyLocationData locationData = https://www.cnblogs.com/1693977889zz/p/locationBuilder.build();
baiduMap.setMyLocationData(locationData);

大體思路就是這個樣子,下面我們開始來實作一下,修改MainActivity中的代碼,如下所示:

public class MainActivity extends AppCompatActivity {
    ...
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //沒有它,會報錯:Please recheck the setAgreePrivacy interface
        mLocationClient.setAgreePrivacy(true);
        try {
            mLocationClient = new LocationClient(getApplicationContext());
        } catch (Exception e) {
            e.printStackTrace();
        }
        mLocationClient.registerLocationListener(new MyLocationListener());
        setContentView(R.layout.activity_main);
        //獲取地圖控制元件參考
        mMapView = (MapView) findViewById(R.id.bmapView);
        baiduMap = mMapView.getMap();
        //根據百度地圖的限制,想要使用這一功能,一定要事先呼叫BaiduMap的setMyLocationEnabled()方法將此功能開啟,否則設備的位置將無法在地圖上顯示,
        baiduMap.setMyLocationEnabled(true);
        ...
    private void navigateTo(BDLocation location) {
        if (isFirstLocate) {
            LatLng ll = new LatLng(location.getLatitude(),location.getLongitude());
            MapStatusUpdate update = MapStatusUpdateFactory.newLatLng(ll);
            baiduMap.animateMapStatus(update);
            update = MapStatusUpdateFactory.zoomTo(16f);
            baiduMap.animateMapStatus(update);
            isFirstLocate = false;
        }
        MyLocationData.Builder locationBuilder = new MyLocationData.Builder();
        locationBuilder.latitude(location.getLatitude());
        locationBuilder.longitude(location.getLongitude());
        MyLocationData locationData = https://www.cnblogs.com/1693977889zz/p/locationBuilder.build();
        baiduMap.setMyLocationData(locationData);
    }
    ...
    @Override
    protected void onDestroy() {
        super.onDestroy();
        //在activity執行onDestroy時執行mMapView.onDestroy(),實作地圖生命周期管理
        mLocationClient.stop();
        mMapView.onDestroy();
        //在程式退出的時候,也要記得將BaiduMap的setMyLocationEnabled()方法功能給關閉掉
        baiduMap.setMyLocationEnabled(false);
    }
}

在navigateTo()方法中,添加了MyLocationData的構建邏輯,將Location中包含的經度和緯度分別封裝到了MyLocationData.Builder當中,最后把MyLocationData設定到了BaiduMap的setMyLocationData()方法當中,

注意:這段邏輯必須寫在isFirstLocate這個if條件陳述句的外面,因為讓地圖移動到我們當前的位置只需要在第一次定位的時候執行,但是設備在地圖上顯示的位置卻應該是隨著設備的移動而實時改變的,

image

想要更加深入地研究百度LBS的各種用法,可以到官方網站上面參考開發指南,地址是:http://lbsyun.baidu.com,另外,百度LBS SDK的版本未來隨時都有可能更新,也許更新之后會導致書上的例子無法正常運行,因此除了照著圖書學習之外,根據官網的開發指南來進行學習也是非常重要的

11.5 Git時間——版本控制工具的高級用法

打開Git Bash,并進入到LBSTest2這個專案的根目錄,然后執行提交操作:

git init
git add
git commit -m "First Commit"

這樣就將準備作業完成了,下面就讓我們開始學習關于Git的高級用法,

11.5.1 分支的用法

分支是版本控制工具中比較高級且比較重要的一個概念,它主要的作用就是在現有代碼的基礎上開辟一個分叉口,使得代碼可以在主干線和分支線上同時進行開發,且相互之間不會影響,分支的作業原理示意圖如圖:

image

為什么需要建立分支呢?只在主干線上進行開發不是挺好的嗎?

通常情況下,只在主干線上進行開發是完全沒有問題的,不過一旦涉及出版本的情況,如果不建立分支的話,你就會非常地頭疼,舉個簡單的例子吧,比如說你們公司研發了一款不錯的軟體,最近剛剛完成,并推出了1.0版本,但是領導是不會讓你們閑著的,馬上提出了新的需求,讓你們投入到了1.1版本的開發作業當中,過了幾個星期,1.1版本的功能已完成了一半,但是這個時候有用戶反饋,之前上線的1.0版本發現了幾個重大的bug,嚴重影響軟體的正常使用,領導也相當重視這個問題,要求你們立刻修復這些bug,并重新發布1.0版本,但這個時候你就非常為難了,你會發現根本沒法去修復這些bug,因為現在1.1版本已開發一半了,如果在現有代碼的基礎上修復這些bug,那么更新的1.0版本將會帶有一半1.1版本的功能!

但是如果你使用了分支的話,就完全不會存在這個讓人頭疼的問題,你只需要在發布1.0版本的時候建立一個分支,然后在主干線上繼續開發1.1版本的功能,當1.0版本上發現任何bug的時候,就在分支線上進行修改,然后發布新的1.0版本,并記得將修改后的代碼合并到主干線上,這樣的話,不僅可以輕松解決掉1.0版本存在的bug,而且保證了主干線上的代碼也已經修復了這些bug,當1.1版本發布時就不會有同樣的bug存在了,

分支的英文名是branch,如果想要查看當前的版本庫當中有哪些分支,可以使用git branch這個命令,結果如圖:

image

由于目前LBSTest2專案中還沒有創建過任何分支,因此只有一個master分支存在,這也就是前面所說的主干線,接下來我們嘗試去創建一個分支,命令如下:

git branch version1.0

這樣就創建了一個名為version1.0的分支,我們再次輸入git branch這個命令來檢查一下,結果如圖:

image

可以看到,果然有一個叫作version1.0的分支出現了,你會發現,master分支的前面有一個“*”號,說明目前我們的代碼還是在master分支上的,怎樣切換到version1.0這個分支上——使用checkout命令即可,如下所示:

git checkout version1.0

再次輸入git branch來進行檢查,結果如圖:

image

可以看到,我們已經把代碼成功切換到version1.0這個分支上了,

需要注意的是,在version1.0分支上修改并提交的代碼將不會影響到master分支,同樣的道理,在master分支上修改并提交的代碼也不會影響到version1.0分支,因此,如果我們在version1.0分支上修復了一個bug,在master分支上這個bug仍然是存在的

這時將修改的代碼一行行復制到master分支上顯然不是一種聰明的做法,最好的辦法就是使用merge命令來完成合并操作,如下所示:

git checkout master
git merge version1.0

僅僅這樣簡單的兩行命令,就可以把在version1.0分支上修改并提交的內容合并到master分支上了,當然,在合并分支的時候還有可能出現代碼沖突的情況,這個時候需要你自己解決這些沖突,Git在這里就無法幫助,

最后,當我們不再需要version1.0這個分支的時候,可以使用如下命令將這個分支洗掉掉:

git branch -D version1.0

image

11.5.2 與遠程版本庫協作

如果你是一個人在開發,那么使用版本控制工具就遠遠無法發揮出它真正強大的功能,

所有版本控制工具最重要的一個特點就是可以使用它來進行團隊合作開發,每個人的電腦上都會有一份代碼,當團隊的某個成員在自己的電腦上撰寫完成了某個功能后,就將代碼提交到服務器,其他的成員只需要將服務器上的代碼同步到本地,就能保證整個團隊所有人的代碼都相同,這樣的話,每個團隊成員就可以各司其職,大家共同來完成一個較為龐大的專案,

那么如何使用Git來進行團隊合作開發呢?這就需要有一個遠程的版本庫,團隊的每個成員都從這個版本庫中獲取到最原始的代碼,然后各自進行開發,并且以后每次提交的代碼都同步到遠程版本庫上就可以了,另外,團隊中的每個成員最好都要養成經常從版本庫中獲取最新代碼的習慣,不然的話,大家的代碼就很有可能經常出現沖突,

比如說現在有一個遠程版本庫的Git地址是https://github.com/example/test.git,就可以使用如下的命令將代碼下載到本地:

git clone https://github.com/example/test.git

之后,你在這份代碼的基礎上進行了一些修改和提交,那么怎樣才能把本地修改的內容同步到遠程版本庫上呢?這就需要借助push命令來完成了,用法如下所示:

git push origin master

其中origin部分指定的是遠程版本庫的Git地址,master部分指定的是同步到哪一個分支上,上述命令就完成了將本地代碼同步到https://github.com/example/test.git這個版本庫的master分支上的功能,

如何將遠程版本庫上的修改同步到本地,Git提供了兩種命令來完成此功能,分別是fetch和pull, fetch的語法規則和push是差不多的,如下所示:

git fetch origin master

執行這個命令后,就會將遠程版本庫上的代碼同步到本地,不過同步下來的代碼并不會合并到任何分支上去,而是會存放到一個origin/master分支上,這時我們可以通過diff命令來查看遠程版本庫上到底修改了哪些東西:

git diff origin/master

之后,再呼叫merge命令將origin/master分支上的修改合并到主分支上即可,如下所示:

git merge origin/master

而pull命令則是相當于將fetch和merge這兩個命令放在一起執行了,它可以從遠程版本庫上獲取最新的代碼并且合并到本地,用法如下所示:

git pull origin/master

11.6 小結與點評

主要學習了基于位置服務的作業原理和用法,借助百度提供的LBS SDK,可以隨時確定自己當前位置的經緯度,并且還能獲取到具體的省、市、區、街道等地址,又學習了百度地圖的用法,不僅成功地將地圖資訊顯示了出來,還綜合利用了前面所學到的定位技術實作了一個較為完整的例子,對Git的用法進行了更深一步的探究,對分支和遠程版本庫的使用都有了一定層次的了解,

下一章中我們將會學習Android 5.0系統中新增的一套全新的知識點——Material Design,

個人學習筆記,針對本人在自學中遇到的問題,

轉載請註明出處,本文鏈接:https://www.uj5u.com/yidong/472411.html

標籤:Android

上一篇:《第一行代碼:Android篇》學習筆記(十)

下一篇:《第一行代碼:Android篇》學習筆記(十二)

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【從零開始擼一個App】Dagger2

    Dagger2是一個IOC框架,一般用于Android平臺,第一次接觸的朋友,一定會被搞得暈頭轉向。它延續了Java平臺Spring框架代碼碎片化,注解滿天飛的傳統。嘗試將各處代碼片段串聯起來,理清思緒,真不是件容易的事。更不用說還有各版本細微的差別。 與Spring不同的是,Spring是通過反射 ......

    uj5u.com 2020-09-10 06:57:59 more
  • Flutter Weekly Issue 66

    新聞 Flutter 季度調研結果分享 教程 Flutter+FaaS一體化任務編排的思考與設計 詳解Dart中如何通過注解生成代碼 GitHub 用對了嗎?Flutter 團隊分享如何管理大型開源專案 插件 flutter-bubble-tab-indicator A Flutter librar ......

    uj5u.com 2020-09-10 06:58:52 more
  • Proguard 常用規則

    介紹 Proguard 入口,如何查看輸出,如何使用 keep 設定入口以及使用實體,如何配置壓縮,混淆,校驗等規則。

    ......

    uj5u.com 2020-09-10 06:59:00 more
  • Android 開發技術周報 Issue#292

    新聞 Android即將獲得類AirDrop功能:可向附近設備快速分享檔案 谷歌為安卓檔案管理應用引入可安全隱藏資料的Safe Folder功能 Android TV新主界面將顯示電影、電視節目和應用推薦內容 泄露的Android檔案暗示了傳說中的谷歌Pixel 5a與折疊屏新機 谷歌發布Andro ......

    uj5u.com 2020-09-10 07:00:37 more
  • AutoFitTextureView Error inflating class

    報錯: Binary XML file line #0: Binary XML file line #0: Error inflating class xxx.AutoFitTextureView 解決: <com.example.testy2.AutoFitTextureView android: ......

    uj5u.com 2020-09-10 07:00:41 more
  • 根據Uri,Cursor沒有獲取到對應的屬性

    Android: 背景:呼叫攝像頭,拍攝視頻,指定保存的地址,但是回傳的Cursor檔案,只有名稱和大小的屬性,沒有其他諸如時長,連ID屬性都沒有 使用 cursor.getInt(cursor.getColumnIndexOrThrow(MediaStore.Video.Media.DURATIO ......

    uj5u.com 2020-09-10 07:00:44 more
  • Android連載29-持久化技術

    一、持久化技術 我們平時所使用的APP產生的資料,在記憶體中都是瞬時的,會隨著斷電、關機等丟失資料,因此android系統采用了持久化技術,用于存盤這些“瞬時”資料 持久化技術包括:檔案存盤、SharedPreference存盤以及資料庫存盤,還有更復雜的SD卡記憶體儲。 二、檔案存盤 最基本存盤方式, ......

    uj5u.com 2020-09-10 07:00:47 more
  • Android Camera2Video整合到自己專案里

    背景: Android專案里呼叫攝像頭拍攝視頻,原本使用的 MediaStore.ACTION_VIDEO_CAPTURE, 后來因專案需要,改成了camera2 1.Camera2Video 官方demo有點問題,下載后,不能直接整合到專案 問題1.多次拍攝視頻崩潰 問題2.雙擊record按鈕, ......

    uj5u.com 2020-09-10 07:00:50 more
  • Android 開發技術周報 Issue#293

    新聞 谷歌為Android TV開發者提供多種新功能 Android 11將自動填表功能整合到鍵盤輸入建議中 谷歌宣布Android Auto即將支持更多的導航和數字停車應用 谷歌Pixel 5只有XL版本 搭載驍龍765G且將比Pixel 4更便宜 [圖]Wear OS將迎來重磅更新:應用啟動時間 ......

    uj5u.com 2020-09-10 07:01:38 more
  • 海豚星空掃碼投屏 Android 接收端 SDK 集成 六步驟

    掃碼投屏,開放網路,獨占設備,不需要額外下載軟體,微信掃碼,發現設備。支持標準DLNA協議,支持倍速播放。視頻,音頻,圖片投屏。好點意思。還支持自定義基于 DLNA 擴展的操作動作。好像要收費,沒體驗。 這里簡單記錄一下集成程序。 一 跟目錄的build.gradle添加私有mevan倉庫 mave ......

    uj5u.com 2020-09-10 07:01:43 more
最新发布
  • 歡迎頁輪播影片

    如圖,引導開始,球從上落下,同時淡入文字,然后文字開始輪播,最后一頁時停止,點擊進入首頁。 在來看看效果圖。 重力球先不講,主要歡迎輪播簡單實作 首先新建一個類 TextTranslationXGuideView,用于影片展示 文本是類似的,最后會有個圖片箭頭影片,布局很簡單,就是一個 TextVi ......

    uj5u.com 2023-04-20 08:40:31 more
  • 【FAQ】關于華為推送服務因營銷訊息頻次管控導致服務通訊類訊息

    一. 問題描述 使用華為推送服務下發IM訊息時,下發訊息請求成功且code碼為80000000,但是手機總是收不到訊息; 在華為推送自助分析(Beta)平臺查看發現,訊息發送觸發了頻控。 二. 問題原因及背景 2023年1月05日起,華為推送服務對咨詢營銷類訊息做了單個設備每日推送數量上限管理,具體 ......

    uj5u.com 2023-04-20 08:40:11 more
  • 歡迎頁輪播影片

    如圖,引導開始,球從上落下,同時淡入文字,然后文字開始輪播,最后一頁時停止,點擊進入首頁。 在來看看效果圖。 重力球先不講,主要歡迎輪播簡單實作 首先新建一個類 TextTranslationXGuideView,用于影片展示 文本是類似的,最后會有個圖片箭頭影片,布局很簡單,就是一個 TextVi ......

    uj5u.com 2023-04-20 08:39:36 more
  • 【FAQ】關于華為推送服務因營銷訊息頻次管控導致服務通訊類訊息

    一. 問題描述 使用華為推送服務下發IM訊息時,下發訊息請求成功且code碼為80000000,但是手機總是收不到訊息; 在華為推送自助分析(Beta)平臺查看發現,訊息發送觸發了頻控。 二. 問題原因及背景 2023年1月05日起,華為推送服務對咨詢營銷類訊息做了單個設備每日推送數量上限管理,具體 ......

    uj5u.com 2023-04-20 08:39:13 more
  • iOS從UI記憶體地址到讀取成員變數(oc/swift)

    開發除錯時,我們發現bug時常首先是從UI顯示發現例外,下一步才會去定位UI相關連的資料的。XCode有給我們提供一系列debug工具,但是很多人可能還沒有形成一套穩定的除錯流程,因此本文嘗試解決這個問題,順便提出一個暴論:UI顯示例外問題只需要兩個步驟就能完成定位作業的80%: 定位例外 UI 組 ......

    uj5u.com 2023-04-19 09:16:23 more
  • FIDE重磅更新!性能飛躍!體驗有禮!

    FIDE 開發者工具重構升級啦!實作500%性能提升,誠邀體驗! 一直以來不少開發者朋友在社區反饋,在使用 FIDE 工具的程序中,時常會遇到諸如加載不及時、代碼預覽/渲染性能不如意的情況,十分影響開發體驗。 作為技術團隊,我們深知一件趁手的開發工具對開發者的重要性,因此,在2023年開年,FinC ......

    uj5u.com 2023-04-19 09:16:15 more
  • 游戲內嵌社區服務開放,助力開發者提升玩家互動與留存

    華為 HMS Core 游戲內嵌社區服務提供快速訪問華為游戲中心論壇能力,支持玩家直接在游戲內瀏覽帖子和交流互動,助力開發者擴展內容生產和觸達的場景。 一、為什么要游戲內嵌社區? 二、游戲內嵌社區的典型使用場景 1、游戲內打開論壇 您可以在游戲內繪制論壇入口,為玩家提供沉浸式發帖、瀏覽、點贊、回帖、 ......

    uj5u.com 2023-04-19 09:15:46 more
  • iOS從UI記憶體地址到讀取成員變數(oc/swift)

    開發除錯時,我們發現bug時常首先是從UI顯示發現例外,下一步才會去定位UI相關連的資料的。XCode有給我們提供一系列debug工具,但是很多人可能還沒有形成一套穩定的除錯流程,因此本文嘗試解決這個問題,順便提出一個暴論:UI顯示例外問題只需要兩個步驟就能完成定位作業的80%: 定位例外 UI 組 ......

    uj5u.com 2023-04-19 09:14:53 more
  • FIDE重磅更新!性能飛躍!體驗有禮!

    FIDE 開發者工具重構升級啦!實作500%性能提升,誠邀體驗! 一直以來不少開發者朋友在社區反饋,在使用 FIDE 工具的程序中,時常會遇到諸如加載不及時、代碼預覽/渲染性能不如意的情況,十分影響開發體驗。 作為技術團隊,我們深知一件趁手的開發工具對開發者的重要性,因此,在2023年開年,FinC ......

    uj5u.com 2023-04-19 09:14:08 more
  • 游戲內嵌社區服務開放,助力開發者提升玩家互動與留存

    華為 HMS Core 游戲內嵌社區服務提供快速訪問華為游戲中心論壇能力,支持玩家直接在游戲內瀏覽帖子和交流互動,助力開發者擴展內容生產和觸達的場景。 一、為什么要游戲內嵌社區? 二、游戲內嵌社區的典型使用場景 1、游戲內打開論壇 您可以在游戲內繪制論壇入口,為玩家提供沉浸式發帖、瀏覽、點贊、回帖、 ......

    uj5u.com 2023-04-19 09:08:34 more