Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> 關於Android編程 >> Android開發中類加載器DexClassLoader的簡單使用講解

Android開發中類加載器DexClassLoader的簡單使用講解

編輯:關於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中的方法。

  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved