標籤:android style blog http io os ar 使用 java
如有轉載,請聲明出處: 時之沙: http://blog.csdn.net/t12x3456 (來自時之沙的csdn部落格)
概述:
隨著應用的不斷迭代,應用的體積不斷增大,項目越來越臃腫,冗餘增加.項目新功能的添加,無法確定與使用者匹配性,發生嚴重異常往往牽一髮而動全身,只能緊急發布補丁版本,強制使用者進行更新.結果頻繁的更新,反而容易降低使用者使用黏性.或者是公司業務的不斷髮展,同系的應用越來越多,傳統方式需要通過使用者量最大的主專案進行引導下載並安裝.
怎麼辦?參考瀏覽器-外掛程式開發模式:
一. 來可以將自己的應用分拆,某些功能可以在外掛程式中實現,用到時再進行下載,而且不用安裝. 如果有新功能的添加,不需要更新應用,只要預留外掛程式管理,我們就可以通過添加外掛程式的方式,動態更新自己的應用,該功能需要改進或擴充,更新外掛程式即可,無需頻繁安裝或卸載(容易造成使用者反感).
二. 對應同系應用,正常的引流方式只能引導使用者進行新應用的下載和安裝,如果使用外掛程式化開發,則無需安裝應用,關閉外掛程式功能也十分方便,省去應用安裝和卸載的過程,可以實現無縫引流.
這裡要向大家推薦一個開源的動態載入架構DL, 該項目由singwhatiwanna發起,目前一共有三個人開發,我有幸成為了其中的contributor.
如果你對DL動態載入架構還不熟悉,建議先看一下這篇文章:
APK動態載入架構DL解析 http://blog.csdn.net/singwhatiwanna/article/details/39937639
如果你看過之後還會不太清晰,請看下DL外掛程式化架構的全景,如下
這裡我主要向大家介紹一下利用DL架構進行開發的具體步驟:
1. 首先我們需要從github上擷取項目代碼:
https://github.com/singwhatiwanna/dynamic-load-apk
這裡我們可以看到下載後的目錄如下
lib目錄就是我們的公用外掛程式庫
sample目錄是對應的demo, 具體的工程可以參照上面的DL全景中的三種模式:
由於一般項目中,代碼管理和Team Dev相對獨立, 一般外掛程式工程的團隊是很難接觸主專案團隊的代碼.因此,這裡主要以第一種,也是我們最為推薦的方式進行, 採用外掛程式不依賴宿主的方式進行開發. 不需要兩個團隊過多的互動,開發效率相對較高.
2 匯入lib工程,如下所示:
可以看到bin目錄下的dl-lib.jar, 如果我們需要對lib工程進行修改,重新build擷取對應的dl-lib.jar即可
3. 外掛程式工程的開發, 匯入demo中的main-plugin工程
首先還是要強調外掛程式開發的注意事項,以免出現不必要的錯誤
外掛程式也需要引用DL的jar包,但是不能放入到外掛程式工程的libs目錄下面,換句話說,就是外掛程式編譯的時候依賴jar包但是打包成apk的時候不要把jar包打進去,這是因為,dl-lib.jar已經在宿主工程中存在了,如果外掛程式中也有這個jar包,就會發生類連結錯誤,原因很簡單,記憶體中有兩份一樣的類,重複了。至於support-v4也是同樣的道理。對於eclipse很簡單,只需要在外掛程式工程中建立一個目錄,比如external-jars,然後把dl-lib.jar和support-v4.jar放進去,同時在.classpath中追加如下兩句即可:
<classpathentry kind="lib" path="external-jars/dl-lib.jar"/><classpathentry kind="lib" path="external-jars/android-support-v4.jar"/>
然後是外掛程式開發中的具體步驟
(1) 如果原有的為Activity,這裡需要改為繼承DLBasePluginActivity,如果原來為FragmentActivity,那麼需要繼承DLBasePluginFragmentActivity, for example:
public class MainActivity extends DLBasePluginActivity
TestFragmentActivity extends DLBasePluginFragmentActivity
(2) 如果需要外掛程式獨立安裝運行, 只要將jar放到libs下面即可,若支援動態載入,仍需按上述注意事項加入 exteral-jars中
(3)外掛程式所需要許可權需要在宿主工程中聲明
如果是實際開發,一般為從伺服器擷取外掛程式,這裡我們方便自己調試示範,將運行產生對應的外掛程式apk,放入sd卡上的DynamicLoadHost目錄中
4. 宿主工程中將上述產生的dl-lib.jar加入libs即可,如下所示為demo中提供的宿主工程
在宿主工程中,首先我們需要擷取要調用的外掛程式apk對應的MainActivity,DL的demo中外掛程式路徑為 sd卡上的DynamicLoadHost目錄,沒有的話需要建立,或者根據自己需求進行修改.
String pluginFolder = Environment.getExternalStorageDirectory() + "/DynamicLoadHost"; File file = new File(pluginFolder); File[] plugins = file.listFiles(); if (plugins == null || plugins.length == 0) { mNoPluginTextView.setVisibility(View.VISIBLE); return; } for (File plugin : plugins) { PluginItem item = new PluginItem(); item.pluginPath = plugin.getAbsolutePath(); item.packageInfo = DLUtils.getPackageInfo(this, item.pluginPath); if (item.packageInfo.activities != null && item.packageInfo.activities.length > 0) { item.launcherActivityName = item.packageInfo.activities[0].name; } mPluginItems.add(item); }
接著是調起響應的apk,這時需要使用dl-lib.jar:
(1)通過Class.forName的方式擷取我們需要調用的外掛程式apk中MainActivity的class對象
(2) 就上面提到的,我們需要判斷該對象繼承自DLProxActivity還是DLProxFragmentActivity,得到對應的代理class對象
(3)使用對應的代理class對象調起外掛程式apk
PluginItem item = mPluginItems.get(position); Class<?> proxyCls = null; try { Class<?> cls = Class.forName(item.launcherActivityName, false, DLClassLoader.getClassLoader(item.pluginPath, getApplicationContext(), getClassLoader())); if (cls.asSubclass(DLBasePluginActivity.class) != null) { proxyCls = DLProxyActivity.class; } } catch (ClassNotFoundException e) { e.printStackTrace(); Toast.makeText(this, "load plugin apk failed, load class " + item.launcherActivityName + " failed.", Toast.LENGTH_SHORT).show(); } catch (ClassCastException e) { // ignored } finally { if (proxyCls == null) { proxyCls = DLProxyFragmentActivity.class; } Intent intent = new Intent(this, proxyCls); intent.putExtra(DLConstants.EXTRA_DEX_PATH, mPluginItems.get(position).pluginPath); startActivity(intent); }
最後運行宿主工程main-host,就可以看到最終的效果:
相信大家經過如上步驟,可以對DL動態載入架構進行開發有了一定瞭解.目前DL架構仍在不斷擴充中,歡迎對我們的項目進行star,fork,或者提出寶貴的建議.,如有問題請及時反饋,我們會在後續版本中進行修複或者改進.
Android 使用動態載入架構DL進行外掛程式化開發