編輯:關於Android編程
之前有很多朋友都問過我,在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()方法執行具體的靜默安裝邏輯。 */ 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 ,-r參數表示如果要安裝的apk已經存在了就覆蓋安裝的意思,apk路徑是作為方法參數傳入的。接下來的幾行就是執行上述命令的過程,注意安裝這個過程是同步的,因此我們在下面調用了process.waitFor()方法,即安裝要多久,我們就要在這裡等多久。等待結束之後說明安裝過程結束了,接下來我們要去讀取安裝的結果並進行解析,解析的邏輯也很簡單,如果安裝結果中包含Failure字樣就說明安裝失敗,反之則說明安裝成功。
<?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>
整個方法還是非常簡單易懂的,下面我們就來搭建調用這個方法的環境。修改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: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>
public class FileExplorerActivity extends AppCompatActivity implements AdapterView.OnItemClickListener { ListView listView; SimpleAdapter adapter; String rootPath = Environment.getExternalStorageDirectory().getPath(); String currentPath = rootPath; List這部分代碼由於和我們本篇文件的主旨沒什麼關系,主要是為了方便demo展示的,因此我就不進行講解了。
接下來修改MainActivity中的代碼,如下所示:
/** * 仿360手機助手秒裝和智能安裝功能的主Activity。 */ 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手機助手,來實現類似的智能安裝功能。
首先在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中指定的內容,如下所示:
接下來修改AndroidManifest.xml文件,在裡面配置無障礙服務: <resources> <string name="app_name">InstallTest</string> <string name="accessibility_service_description">智能安裝服務,無需用戶的任何操作就可以自動安裝程序。</string> </resources>智能安裝服務,無需用戶的任何操作就可以自動安裝程序。
<?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>
接下來就是要去實現智能安裝功能的具體邏輯了,創建一個MyAccessibilityService類並繼承自AccessibilityService,代碼如下所示:
/** * 智能安裝功能的實現類。 * @since 2015/12/7 */ public class MyAccessibilityService extends AccessibilityService { Map代碼並不復雜,我們來解析一下。每當窗口有活動時,就會有消息回調到onAccessibilityEvent()方法中,因此所有的邏輯都是從這裡開始的。首先我們可以通過傳入的AccessibilityEvent參數來獲取當前事件的類型,事件的種類非常多,但是我們只需要監聽TYPE_WINDOW_CONTENT_CHANGED和TYPE_WINDOW_STATE_CHANGED這兩種事件就可以了,因為在整個安裝過程中,這兩個事件必定有一個會被觸發。當然也有兩個同時都被觸發的可能,那麼為了防止二次處理的情況,這裡我們使用了一個Map來過濾掉重復事件。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() { } }
接下來就是調用iterateNodesAndHandle()方法來去解析當前界面的節點了,這裡我們通過遞歸的方式將安裝界面中所有的子節點全部進行遍歷,當發現按鈕節點的時候就進行判斷,按鈕上的文字是不是“安裝”、“完成”、“確定”這幾種類型,如果是的話就模擬一下點擊事件,這樣也就相當於幫用戶自動操作了這些按鈕。另外從Android 4.4系統開始,用戶需要將應用申請的所有權限看完才可以點擊安裝,因此如果我們在節點中發現了ScrollView,那就模擬一下滑動事件,將界面滑動到最底部,這樣安裝按鈕就可以點擊了。
最後,回到MainActivity中,來增加對智能安裝功能的調用,如下所示:
/** * 仿360手機助手秒裝和智能安裝功能的主Activity。 */ 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手機助手也只能實現到這一步而已,那些被產品經理逼著去實現靜默安裝的程序員們也有理由交差了吧?
首先帶大家看一下實現效果,用了兩種實現方式:1.基於LinearLayout實現,導航欄不可響應手指滑動2.基於HorizontalScrollView實現,導航欄可響應
寫了一個月應用層代碼,感覺寫嘔了,最近在研究插件化動態加載方面的東西。本文需要解決的作業:在Activity自身的跳轉中進行Hook。先簡要說下遇到的幾個坑以及後面的學習
上一節,已經完成了來電管家的界面設計,那麼下面就要實現具體的功能了,如何將添加的黑白名單顯示呢?這裡用到了ListView,那麼,如果需要刪除黑白名單呢,是一個個長按彈出
Android特效專輯(九)——仿微信雷達搜索好友特效,邏輯清晰實現簡單 不知不覺這個春節也已經過完了,遺憾家裡沒網,沒能及時給大家送上祝福,今天