編輯:關於Android編程
簡介
“類裝載器”(ClassLoader),顧名思義,就是用來動態裝載class文件的。標准的Java SDK中有個ClassLoader類,借助此類可以裝載需要的class文件,前提是ClassLoader類初始化必須制定class文件的路徑。
import關鍵字引用的類文件和ClassLoader動態加載類的區別:
import引用類的兩個特點:
1、必須存在於本地,當程序運行該類時,內部類裝載器會自動裝載該類。
2、編譯時必須在現場,否則編譯過程會因找不到引用文件而不能正常編譯。
classLoader的特點正好於import相反,而且更自由靈活。
每一個ClassLoader必須有一個父ClassLoader,在裝載Class文件時,子ClassLoader會先請求其父ClassLoader加載該文件,只有當其父ClassLoader找不到該文件時,子ClassLoader才會繼承裝載該類。這是一種安全機制。對於Android而言,最終的apk文件包含的是dex類型的文件,dex文件是將class文件重新打包,打包的規則又不是簡單地壓縮,而是完全對class文件內部的各種函數表,變量表進行優化,產生一個新的文件,即dex文件。因此加載這種特殊的Class文件就需要特殊的類加載器DexClassLoader。
在Java中涉及到的類加載器就是ClassLoader這個類,通過ClassLoader.forName()的方法可以加載我們需要的類,從而實現在運行時動態加載類庫的需求。但是在android中直接使用ClassLoader是行不通的,因為ClassLoader加載的java的字節碼文件,而在android中使用的是dex格式的字節碼,對此android專門提供了一個DexClassLoader類來完成動態加載apk的需求。
實例
下面用一個簡單的例子來說明一下DexClassLoader這個類的使用,這個例子涉及到兩個知識點:跨包取資源 & 反射調用方法。 首先建立一個Client工程,這個工程很簡單,只有一個簡單的layout和一個sayHello()的方法。
public class Main extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); this.setContentView(R.layout.main); } @SuppressWarnings("unused") private void sayHello(String msg){ Log.d("mmtag",msg); } }
接著建立另外一個工程,在這個工程中調用client工程中的view和調用sayHello()方法。在這個類中主要涉及到了DexClassLoader這個類的使用。
package com.example.dexclassloaderserver; import java.lang.reflect.Method; import java.util.Collections; import java.util.List; import dalvik.system.DexClassLoader; import android.annotation.SuppressLint; import android.app.Activity; import android.content.Intent; import android.content.pm.ActivityInfo; import android.content.pm.PackageManager; import android.content.pm.ResolveInfo; import android.os.Bundle; import android.util.Log; public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); useDexClassLoader(); } @SuppressLint("NewApi") public void useDexClassLoader() { Intent mIntent = new Intent(); mIntent.setClassName("com.example.dexclassloaderclient", "com.example.dexclassloaderclient.MainActivity"); PackageManager pm = this.getPackageManager(); List<ResolveInfo> mList = pm.queryIntentActivities(mIntent, PackageManager.MATCH_DEFAULT_ONLY); ResolveInfo info = mList.get(0); String apkPath = info.activityInfo.applicationInfo.sourceDir; String optPath = this.getCodeCacheDir().getAbsolutePath(); String libPath = info.activityInfo.applicationInfo.nativeLibraryDir; DexClassLoader clsLoader = new DexClassLoader(apkPath, optPath, libPath, this.getClass().getClassLoader()); try { Class cls = clsLoader .loadClass("com.example.dexclassloaderclient.MainActivity"); Object obj = cls.newInstance(); Method invokeMethod = cls.getDeclaredMethod("sayHello", new Class[] { String.class }); invokeMethod.setAccessible(true); invokeMethod.invoke(obj, "hello world,DexClassLoader"); } catch (Exception e) { e.printStackTrace(); } } }
生成的結果為:
I/dex2oat (20250): dex2oat took 1.547s (threads: 4) D/mmtag (20229): hello world,DexClassLoader D/OpenGLRenderer(20229): Render dirty regions requested: tru
這樣就成功的調用了其他的apk中的方法。
首先我們先看下效果圖實現思路這是兩張前後對比圖,右邊第二張圖裡面的已搶光標簽圖片當已經沒有商品的時候就會顯示了,在每個圖片的中心位置,第一想法是在ImageView的外層
觀察者設計模式的定義:定義了對象之間的一對多依賴,這樣一來,當一個對象改變狀態時,它的所有依賴者都會收到通知並自動更新。 面向對象設計原則:為交互對象之間的松耦全設計而努
何為動態加載、插件化?需求驅動隨著業務發展需要和無線部門的拆分,各業務產品模塊歸屬到各業務BU,原有無線App開發團隊被分為基礎框架、業務A、業務B、業務C等多個開發團隊
在Android系統自帶的設置功能裡,進入應用列表後,點擊列表條目會進入應用的詳情界面。那麼我們自己可不可以寫一個自己的應用程序,實現激活應用程序的詳情界面呢?這個是可以