產品經理總是有很多反人類的想法(算不上,但是一直很刁鑽……),比如讓咱實現 [靜默安裝] 什麼的(難搞哦),這裏向大家推薦一個來自郭霖大佬的仿360手機助手秒裝和智能安裝功能實現的分享。
原文地址:https://juejin.cn/post/6971732955945238559
希望對大家的學習工作有所啓發和幫助。
前言
之前有很多朋友都問過我,在 Android 系統中怎樣才能實現靜默安裝呢?所謂的靜默安裝,就是不用彈出系統的安裝界面,在不影響用戶任何操作的情況下不知不覺地將程序裝好。
雖說這種方式看上去不打攪用戶,但是卻存在着一個問題,因爲 Android 系統會在安裝界面當中把程序所聲明的權限展示給用戶看,用戶來評估一下這些權限然後決定是否要安裝該程序,但如果使用了靜默安裝的方式,也就沒有地方讓用戶看權限了,相當於用戶被動接受了這些權限。
在 Android 官方看來,這顯示是一種非常危險的行爲,因此靜默安裝這一行爲系統是不會開放給開發者的。
但是總是彈出一個安裝對話框確實是一種體驗比較差的行爲,這一點 Google 自己也意識到了,因此 Android 系統對自家的 Google Play 商店開放了靜默安裝權限,也就是說所有從 Google Play 上下載的應用都可以不用彈出安裝對話框了。這一點充分說明了擁有權限的重要性,自家的系統想怎麼改就怎麼改。
借鑑 Google 的做法,很多國內的手機廠商也採用了類似的處理方式,比如說小米手機在小米商店中下載應用也是不需要彈出安裝對話框的,因爲小米可以在 MIUI 中對 Android 系統進行各種定製。因此,如果我們只是做一個普通的應用,其實不太需要考慮靜默安裝這個功能,因爲我們只需要將應用上架到相應的商店當中,就會自動擁有靜默安裝的功能。
但是如果我們想要做的也是一個類似於商店的平臺呢?比如說像 360 手機助手,它廣泛安裝於各種各樣的手機上,但都是作爲一個普通的應用存在的,而沒有 Google 或小米這樣的特殊權限,那 360 手機助手應該怎樣做到更好的安裝體驗呢?爲此 360 手機助手提供了兩種方案, 秒裝(需 ROOT 權限)和智能安裝,如下圖示:
因此,今天我們就模仿一下 360 手機助手的實現方式,來給大家提供一套靜默安裝的解決方案。
一、秒裝
所謂的秒裝其實就是需要 ROOT 權限的靜默安裝,其實靜默安裝的原理很簡單,就是調用 Android 系統的 pm install 命令就可以了,但關鍵的問題就在於,pm 命令系統是不授予我們權限調用的,因此只能在擁有 ROOT 權限的手機上去申請權限纔行。
下面我們開始動手,新建一個 InstallTest 項目,然後創建一個 SilentInstall 類作爲靜默安裝功能的實現類,代碼如下所示:
/**
* 靜默安裝的實現類,調用install()方法執行具體的靜默安裝邏輯。
* 原文地址:http://blog.csdn.net/guolin_blog/article/details/47803149
* @author guolin
* @since 2015/12/7
*/
public class SilentInstall {
/**
* 執行具體的靜默安裝邏輯,需要手機ROOT。
* @param apkPath
* 要安裝的apk文件的路徑
* @return 安裝成功返回true,安裝失敗返回false。
*/
public boolean install(String apkPath) {
boolean result = false;
DataOutputStream dataOutputStream = null;
BufferedReader errorStream = null;
try {
// 申請su權限
Process process = Runtime.getRuntime().exec("su");
dataOutputStream = new DataOutputStream(process.getOutputStream());
// 執行pm install命令
String command = "pm install -r " + apkPath + "\n";
dataOutputStream.write(command.getBytes(Charset.forName("utf-8")));
dataOutputStream.flush();
dataOutputStream.writeBytes("exit\n");
dataOutputStream.flush();
process.waitFor();
errorStream = new BufferedReader(new InputStreamReader(process.getErrorStream()));
String msg = "";
String line;
// 讀取命令的執行結果
while ((line = errorStream.readLine()) != null) {
msg += line;
}
Log.d("TAG", "install msg is " + msg);
// 如果執行結果中包含Failure字樣就認爲是安裝失敗,否則就認爲安裝成功
if (!msg.contains("Failure")) {
result = true;
}
} catch (Exception e) {
Log.e("TAG", e.getMessage(), e);
} finally {
try {
if (dataOutputStream != null) {
dataOutputStream.close();
}
if (errorStream != null) {
errorStream.close();
}
} catch (IOException e) {
Log.e("TAG", e.getMessage(), e);
}
}
return result;
}
}
可以看到,SilentInstall 類中只有一個 install() 方法,所有靜默安裝的邏輯都在這個方法中了,那麼我們具體來看一下這個方法。首先在第 21 行調用了 Runtime.getRuntime().exec("su") 方法,在這裏先申請 ROOT 權限,不然的話後面的操作都將失敗。然後在第 24 行開始組裝靜默安裝命令,命令的格式就是 pm install -r <apk 路徑 >,-r 參數表示如果要安裝的 apk 已經存在了就覆蓋安裝的意思,apk 路徑是作爲方法參數傳入的。接下來的幾行就是執行上述命令的過程,注意安裝這個過程是同步的,因此我們在下面調用了 process.waitFor() 方法,即安裝要多久,我們就要在這裏等多久。等待結束之後說明安裝過程結束了,接下來我們要去讀取安裝的結果並進行解析,解析的邏輯也很簡單,如果安裝結果中包含 Failure 字樣就說明安裝失敗,反之則說明安裝成功。
整個方法還是非常簡單易懂的,下面我們就來搭建調用這個方法的環境。修改 activity_main.xml 中的代碼,如下所示:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="com.example.installtest.MainActivity">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content">
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="onChooseApkFile"
android:text="選擇安裝包" />
<TextView
android:id="@+id/apkPathText"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:layout_gravity="center_vertical"
/>
</LinearLayout>
<View
android:layout_width="match_parent"
android:layout_height="1dp"
android:background="@android:color/darker_gray" />
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="onSilentInstall"
android:text="秒裝" />
<View
android:layout_width="match_parent"
android:layout_height="1dp"
android:background="@android:color/darker_gray" />
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="onForwardToAccessibility"
android:text="開啓智能安裝服務" />
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="onSmartInstall"
android:text="智能安裝" />
</LinearLayout>
這裏我們先將程序的主界面確定好,主界面上擁有四個按鈕,第一個按鈕用於選擇 apk 文件的,第二個按鈕用於開始秒裝,第三個按鈕用於開啓智能安裝服務,第四個按鈕用於開始智能安裝,這裏我們暫時只能用到前兩個按鈕。那麼調用 SilentInstall 的 install() 方法需要傳入 apk 路
徑,因此我們需要先把文件選擇器的功能實現好,新建 activity_file_explorer.xml 和 list_item.xml 作爲文件選擇器的佈局文件,代碼分別如下所示:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ListView
android:id="@+id/list_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
/>
</LinearLayout>
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:padding="4dp"
android:orientation="horizontal">
<ImageView android:id="@+id/img"
android:layout_width="32dp"
android:layout_margin="4dp"
android:layout_gravity="center_vertical"
android:layout_height="32dp"/>
<TextView android:id="@+id/name"
android:textSize="18sp"
android:textStyle="bold"
android:layout_width="match_parent"
android:gravity="center_vertical"
android:layout_height="50dp"/>
</LinearLayout>
然後新建 FileExplorerActivity 作爲文件選擇器的 Activity,代碼如下:
public class FileExplorerActivity extends AppCompatActivity implements AdapterView.OnItemClickListener {
ListView listView;
SimpleAdapter adapter;
String rootPath = Environment.getExternalStorageDirectory().getPath();
String currentPath = rootPath;
List<Map<String, Object>> list = new ArrayList<>();
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_file_explorer);
listView = (ListView) findViewById(R.id.list_view);
adapter = new SimpleAdapter(this, list, R.layout.list_item,
new String[]{"name", "img"}, new int[]{R.id.name, R.id.img});
listView.setAdapter(adapter);
listView.setOnItemClickListener(this);
refreshListItems(currentPath);
}
private void refreshListItems(String path) {
setTitle(path);
File[] files = new File(path).listFiles();
list.clear();
if (files != null) {
for (File file : files) {
Map<String, Object> map = new HashMap<>();
if (file.isDirectory()) {
map.put("img", R.drawable.directory);
} else {
map.put("img", R.drawable.file_doc);
}
map.put("name", file.getName());
map.put("currentPath", file.getPath());
list.add(map);
}
}
adapter.notifyDataSetChanged();
}
@Override
public void onItemClick(AdapterView<?> parent, View v, int position, long id) {
currentPath = (String) list.get(position).get("currentPath");
File file = new File(currentPath);
if (file.isDirectory())
refreshListItems(currentPath);
else {
Intent intent = new Intent();
intent.putExtra("apk_path", file.getPath());
setResult(RESULT_OK, intent);
finish();
}
}
@Override
public void onBackPressed() {
if (rootPath.equals(currentPath)) {
super.onBackPressed();
} else {
File file = new File(currentPath);
currentPath = file.getParentFile().getPath();
refreshListItems(currentPath);
}
}
}
這部分代碼由於和我們本篇文件的主旨沒什麼關係,主要是爲了方便 demo 展示的,因此我就不進行講解了。
接下來修改 MainActivity 中的代碼,如下所示:
/**
* 仿360手機助手秒裝和智能安裝功能的主Activity。
* 原文地址:http://blog.csdn.net/guolin_blog/article/details/47803149
* @author guolin
* @since 2015/12/7
*/
public class MainActivity extends AppCompatActivity {
TextView apkPathText;
String apkPath;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
apkPathText = (TextView) findViewById(R.id.apkPathText);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == 0 && resultCode == RESULT_OK) {
apkPath = data.getStringExtra("apk_path");
apkPathText.setText(apkPath);
}
}
public void onChooseApkFile(View view) {
Intent intent = new Intent(this, FileExplorerActivity.class);
startActivityForResult(intent, 0);
}
public void onSilentInstall(View view) {
if (!isRoot()) {
Toast.makeText(this, "沒有ROOT權限,不能使用秒裝", Toast.LENGTH_SHORT).show();
return;
}
if (TextUtils.isEmpty(apkPath)) {
Toast.makeText(this, "請選擇安裝包!", Toast.LENGTH_SHORT).show();
return;
}
final Button button = (Button) view;
button.setText("安裝中");
new Thread(new Runnable() {
@Override
public void run() {
SilentInstall installHelper = new SilentInstall();
final boolean result = installHelper.install(apkPath);
runOnUiThread(new Runnable() {
@Override
public void run() {
if (result) {
Toast.makeText(MainActivity.this, "安裝成功!", Toast.LENGTH_SHORT).show();
} else {
Toast.makeText(MainActivity.this, "安裝失敗!", Toast.LENGTH_SHORT).show();
}
button.setText("秒裝");
}
});
}
}).start();
}
public void onForwardToAccessibility(View view) {
}
public void onSmartInstall(View view) {
}
/**
* 判斷手機是否擁有Root權限。
* @return 有root權限返回true,否則返回false。
*/
public boolean isRoot() {
boolean bool = false;
try {
bool = new File("/system/bin/su").exists() || new File("/system/xbin/su").exists();
} catch (Exception e) {
e.printStackTrace();
}
return bool;
}
}
可以看到,在 MainActivity 中,我們對四個按鈕點擊事件的回調方法都進行了定義,當點擊選擇安裝包按鈕時就會調用 onChooseApkFile() 方法,當點擊秒裝按鈕時就會調用 onSilentInstall() 方法。在 onChooseApkFile() 方法方法中,我們通過 Intent 打開了 FileExplorerActivity,然後在 onActivityResult() 方法當中讀取選擇的 apk 文件路徑。在 onSilentInstall() 方法當中,先判斷設備是否 ROOT,如果沒有 ROOT 就直接 return,然後判斷安裝包是否已選擇,如果沒有也直接 return。接下來我們開啓了一個線程來調用 SilentInstall.install() 方法,因爲安裝過程會比較耗時,如果不開線程的話主線程就會被卡住,不管安裝成功還是失敗,最後都會使用 Toast 來進行提示。
代碼就這麼多,最後我們來配置一下 AndroidManifest.xml 文件:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.installtest">
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity android:name=".FileExplorerActivity"/>
</application>
</manifest>
並沒有什麼特殊的地方,由於選擇 apk 文件需要讀取 SD 卡,因此在 AndroidManifest.xml 文件中要記得聲明讀 SD 卡權限。
另外還有一點需要注意,在 Android 6.0 系統中,讀寫 SD 卡權限被列爲了危險權限,因此如果將程序的 targetSdkVersion 指定成了 23 則需要做專門的 6.0 適配,這裏簡單起見,我把 targetSdkVersion 指定成了 22,因爲 6.0 的適配工作也不在文章的講解範圍之內。
現在運行程序,就可以來試一試秒裝功能了,切記手機一定要 ROOT,效果如下圖所示:
可以看到,這裏我們選擇的網易新聞安裝包已成功安裝到手機上了,並且沒有彈出系統的安裝界面,由此證明秒裝功能已經成功實現了。
二、智能安裝
那麼對於 ROOT 過的手機,秒裝功能確實可以避免彈出系統安裝界面,在不影響用戶操作的情況下實現靜默安裝,但是對於絕大部分沒有 ROOT 的手機,這個功能是不可用的。那麼我們應該怎麼辦呢?爲此 360 手機助手提供了一種折中方案,就是藉助 Android 提供的無障礙服務來實現智能安裝。所謂的智能安裝其實並不是真正意義上的靜默安裝,因爲它還是要彈出系統安裝界面的,只不過可以在安裝界面當中釋放用戶的操作,由智能安裝功能來模擬用戶點擊,安裝完成之後自動關閉界面。這個功能是需要用戶手動開啓的,並且只支持 Android 4.1 之後的手機,如下圖所示:
好的,那麼接下來我們就模仿一下 360 手機助手,來實現類似的智能安裝功能。
智能安裝功能的實現原理要藉助 Android 提供的無障礙服務,關於無障礙服務的詳細講解可參考官方文檔:developer.android.com/guide/topic…。
首先在 res/xml 目錄下新建一個 accessibility_service_config.xml 文件,代碼如下所示:
<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"
android:packageNames="com.android.packageinstaller"
android:description="@string/accessibility_service_description"
android:accessibilityEventTypes="typeAllMask"
android:accessibilityFlags="flagDefault"
android:accessibilityFeedbackType="feedbackGeneric"
android:canRetrieveWindowContent="true"
/>
其中,packageNames 指定我們要監聽哪個應用程序下的窗口活動,這裏寫 com.android.packageinstaller 表示監聽 Android 系統的安裝界面。description 指定在無障礙服務當中顯示給用戶看的說明信息,上圖中 360 手機助手的一大段內容就是在這裏指定的。accessibilityEventTypes 指定我們在監聽窗口中可以模擬哪些事件,這裏寫 typeAllMask 表示所有的事件都能模擬。accessibilityFlags 可以指定無障礙服務的一些附加參數,這裏我們傳默認值 flagDefault 就行。accessibilityFeedbackType 指定無障礙服務的反饋方式,實際上無障礙服務這個功能是 Android 提供給一些殘疾人士使用的,比如說盲人不方便使用手機,就可以藉助無障礙服務配合語音反饋來操作手機,而我們其實是不需要反饋的,因此隨便傳一個值就可以,這裏傳入 feedbackGeneric。最後 canRetrieveWindowContent 指定是否允許我們的程序讀取窗口中的節點和內容,必須寫 true。
記得在 string.xml 文件中寫一下 description 中指定的內容,如下所示:
<resources>
<string name="app_name">InstallTest</string>
<string name="accessibility_service_description">智能安裝服務,無需用戶的任何操作就可以自動安裝程序。</string>
</resources>
接下來修改 AndroidManifest.xml 文件,在裏面配置無障礙服務:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.installtest">
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
......
<service
android:name=".MyAccessibilityService"
android:label="我的智能安裝"
android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE">
<intent-filter>
<action android:name="android.accessibilityservice.AccessibilityService" />
</intent-filter>
<meta-data
android:name="android.accessibilityservice"
android:resource="@xml/accessibility_service_config" />
</service>
</application>
</manifest>
這部分配置的內容多數是固定的,必須要聲明一個 android.permission.BIND_ACCESSIBILITY_SERVICE 的權限,且必須要有一個值爲 android.accessibilityservice.AccessibilityService 的 action,然後我們通過 將剛纔創建的配置文件指定進去。
接下來就是要去實現智能安裝功能的具體邏輯了,創建一個 MyAccessibilityService 類並繼承自 AccessibilityService,代碼如下所示:
/**
* 智能安裝功能的實現類。
* 原文地址:http://blog.csdn.net/guolin_blog/article/details/47803149
* @author guolin
* @since 2015/12/7
*/
public class MyAccessibilityService extends AccessibilityService {
Map<Integer, Boolean> handledMap = new HashMap<>();
public MyAccessibilityService() {
}
@Override
public void onAccessibilityEvent(AccessibilityEvent event) {
AccessibilityNodeInfo nodeInfo = event.getSource();
if (nodeInfo != null) {
int eventType = event.getEventType();
if (eventType== AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED ||
eventType == AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED) {
if (handledMap.get(event.getWindowId()) == null) {
boolean handled = iterateNodesAndHandle(nodeInfo);
if (handled) {
handledMap.put(event.getWindowId(), true);
}
}
}
}
}
private boolean iterateNodesAndHandle(AccessibilityNodeInfo nodeInfo) {
if (nodeInfo != null) {
int childCount = nodeInfo.getChildCount();
if ("android.widget.Button".equals(nodeInfo.getClassName())) {
String nodeContent = nodeInfo.getText().toString();
Log.d("TAG", "content is " + nodeContent);
if ("安裝".equals(nodeContent)
|| "完成".equals(nodeContent)
|| "確定".equals(nodeContent)) {
nodeInfo.performAction(AccessibilityNodeInfo.ACTION_CLICK);
return true;
}
} else if ("android.widget.ScrollView".equals(nodeInfo.getClassName())) {
nodeInfo.performAction(AccessibilityNodeInfo.ACTION_SCROLL_FORWARD);
}
for (int i = 0; i < childCount; i++) {
AccessibilityNodeInfo childNodeInfo = nodeInfo.getChild(i);
if (iterateNodesAndHandle(childNodeInfo)) {
return true;
}
}
}
return false;
}
@Override
public void onInterrupt() {
}
}
代碼並不複雜,我們來解析一下。每當窗口有活動時,就會有消息回調到 onAccessibilityEvent() 方法中,因此所有的邏輯都是從這裏開始的。首先我們可以通過傳入的 AccessibilityEvent 參數來獲取當前事件的類型,事件的種類非常多,但是我們只需要監聽 TYPE_WINDOW_CONTENT_CHANGED 和 TYPE_WINDOW_STATE_CHANGED 這兩種事件就可以了,因爲在整個安裝過程中,這兩個事件必定有一個會被觸發。當然也有兩個同時都被觸發的可能,那麼爲了防止二次處理的情況,這裏我們使用了一個 Map 來過濾掉重複事件。
接下來就是調用 iterateNodesAndHandle()方法來去解析當前界面的節點了,這裏我們通過遞歸的方式將安裝界面中所有的子節點全部進行遍歷,當發現按鈕節點的時候就進行判斷,按鈕上的文字是不是 “安裝”、“完成”、“確定” 這幾種類型,如果是的話就模擬一下點擊事件,這樣也就相當於幫用戶自動操作了這些按鈕。另外從 Android 4.4 系統開始,用戶需要將應用申請的所有權限看完纔可以點擊安裝,因此如果我們在節點中發現了 ScrollView,那就模擬一下滑動事件,將界面滑動到最底部,這樣安裝按鈕就可以點擊了。
最後,回到 MainActivity 中,來增加對智能安裝功能的調用,如下所示:
/**
* 仿360手機助手秒裝和智能安裝功能的主Activity。
* 原文地址:http://blog.csdn.net/guolin_blog/article/details/47803149
* @author guolin
* @since 2015/12/7
*/
public class MainActivity extends AppCompatActivity {
......
public void onForwardToAccessibility(View view) {
Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);
startActivity(intent);
}
public void onSmartInstall(View view) {
if (TextUtils.isEmpty(apkPath)) {
Toast.makeText(this, "請選擇安裝包!", Toast.LENGTH_SHORT).show();
return;
}
Uri uri = Uri.fromFile(new File(apkPath));
Intent localIntent = new Intent(Intent.ACTION_VIEW);
localIntent.setDataAndType(uri, "application/vnd.android.package-archive");
startActivity(localIntent);
}
}
當點擊了開啓智能安裝服務按鈕時,我們通過 Intent 跳轉到系統的無障礙服務界面,在這裏啓動智能安裝服務。當點擊了智能安裝按鈕時,我們通過 Intent 跳轉到系統的安裝界面,之後所有的安裝操作都會自動完成了。
現在可以重新運行一下程序,效果如下圖所示:
可以看到,當打開網易新聞的安裝界面之後,我們不需要進行任何的手動操作,界面的滑動、安裝按鈕、完成按鈕的點擊都是自動完成的,最終會自動回到手機原來的界面狀態,這就是仿照 360 手機助手實現的智能安裝功能。
好的,本篇文章的所有內容就到這裏了,雖說不能說完全實現靜默安裝,但是我們已經在權限允許的範圍內儘可能地去完成了,並且 360 手機助手也只能實現到這一步而已,那些被產品經理逼着去實現靜默安裝的程序員們也有理由交差了吧?
最後
需要項目實戰視頻的可以去關注這個B站號:Android開發駱駝
這裏有很多項目實戰還有很多源碼分析的視頻。
B站系列視頻:
觀看記得點贊哦~