Android開(kāi)發(fā)中類(lèi)加載器DexClassLoader的簡(jiǎn)單使用講解
簡(jiǎn)介
“類(lèi)裝載器”(ClassLoader),顧名思義,就是用來(lái)動(dòng)態(tài)裝載class文件的。標(biāo)準(zhǔn)的Java SDK中有個(gè)ClassLoader類(lèi),借助此類(lèi)可以裝載需要的class文件,前提是ClassLoader類(lèi)初始化必須制定class文件的路徑。
import關(guān)鍵字引用的類(lèi)文件和ClassLoader動(dòng)態(tài)加載類(lèi)的區(qū)別:
import引用類(lèi)的兩個(gè)特點(diǎn):
1、必須存在于本地,當(dāng)程序運(yùn)行該類(lèi)時(shí),內(nèi)部類(lèi)裝載器會(huì)自動(dòng)裝載該類(lèi)。
2、編譯時(shí)必須在現(xiàn)場(chǎng),否則編譯過(guò)程會(huì)因找不到引用文件而不能正常編譯。
classLoader的特點(diǎn)正好于import相反,而且更自由靈活。
每一個(gè)ClassLoader必須有一個(gè)父ClassLoader,在裝載Class文件時(shí),子ClassLoader會(huì)先請(qǐng)求其父ClassLoader加載該文件,只有當(dāng)其父ClassLoader找不到該文件時(shí),子ClassLoader才會(huì)繼承裝載該類(lèi)。這是一種安全機(jī)制。對(duì)于Android而言,最終的apk文件包含的是dex類(lèi)型的文件,dex文件是將class文件重新打包,打包的規(guī)則又不是簡(jiǎn)單地壓縮,而是完全對(duì)class文件內(nèi)部的各種函數(shù)表,變量表進(jìn)行優(yōu)化,產(chǎn)生一個(gè)新的文件,即dex文件。因此加載這種特殊的Class文件就需要特殊的類(lèi)加載器DexClassLoader。
在Java中涉及到的類(lèi)加載器就是ClassLoader這個(gè)類(lèi),通過(guò)ClassLoader.forName()的方法可以加載我們需要的類(lèi),從而實(shí)現(xiàn)在運(yùn)行時(shí)動(dòng)態(tài)加載類(lèi)庫(kù)的需求。但是在android中直接使用ClassLoader是行不通的,因?yàn)镃lassLoader加載的java的字節(jié)碼文件,而在android中使用的是dex格式的字節(jié)碼,對(duì)此android專(zhuān)門(mén)提供了一個(gè)DexClassLoader類(lèi)來(lái)完成動(dòng)態(tài)加載apk的需求。
實(shí)例
下面用一個(gè)簡(jiǎn)單的例子來(lái)說(shuō)明一下DexClassLoader這個(gè)類(lèi)的使用,這個(gè)例子涉及到兩個(gè)知識(shí)點(diǎn):跨包取資源 & 反射調(diào)用方法。 首先建立一個(gè)Client工程,這個(gè)工程很簡(jiǎn)單,只有一個(gè)簡(jiǎn)單的layout和一個(gè)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);
}
}
接著建立另外一個(gè)工程,在這個(gè)工程中調(diào)用client工程中的view和調(diào)用sayHello()方法。在這個(gè)類(lèi)中主要涉及到了DexClassLoader這個(gè)類(lèi)的使用。
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();
}
}
}
生成的結(jié)果為:
I/dex2oat (20250): dex2oat took 1.547s (threads: 4) D/mmtag (20229): hello world,DexClassLoader D/OpenGLRenderer(20229): Render dirty regions requested: tru
這樣就成功的調(diào)用了其他的apk中的方法。
相關(guān)文章
Android 開(kāi)發(fā)中根據(jù)搜索內(nèi)容實(shí)現(xiàn)TextView中的文字部分加粗
最近遇到一個(gè)需求,需要做一個(gè)搜索功能。搜索的內(nèi)容需要加粗顯示。實(shí)現(xiàn)方法很簡(jiǎn)單,下面通過(guò)本文給大家分享Android 開(kāi)發(fā)中根據(jù)搜索內(nèi)容實(shí)現(xiàn)TextView中的文字部分加粗樣式,非常不錯(cuò),需要的朋友參考下2017-03-03
Android setTag方法的key問(wèn)題解決辦法
這篇文章主要介紹了Android setTag方法的key問(wèn)題解決辦法的相關(guān)資料,需要的朋友可以參考下2016-09-09
Android通過(guò)overScrollBy實(shí)現(xiàn)下拉視差特效
這篇文章主要為大家詳細(xì)介紹了Android通過(guò)overScrollBy實(shí)現(xiàn)下拉視差特效,實(shí)現(xiàn)精彩的阻尼效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08
android中Webview實(shí)現(xiàn)截屏三種方式小結(jié)
本篇文章主要介紹了android Webview實(shí)現(xiàn)截屏,主要詳解了3種方式,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-03-03
EditText實(shí)現(xiàn)輸入限制和校驗(yàn)功能實(shí)例代碼
本文通過(guò)實(shí)例代碼給大家介紹EditText實(shí)現(xiàn)輸入限制和校驗(yàn)功能,感興趣的朋友參考下吧2017-08-08
Android實(shí)現(xiàn)美團(tuán)外賣(mài)底部導(dǎo)航欄動(dòng)畫(huà)
這篇文章主要介紹了Android實(shí)現(xiàn)美團(tuán)外賣(mài)底部導(dǎo)航欄動(dòng)畫(huà),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11
Android 矢量室內(nèi)地圖開(kāi)發(fā)實(shí)例
這篇文章主要介紹了Android 矢量室內(nèi)地圖開(kāi)發(fā)實(shí)例的相關(guān)資料,這里提供代碼實(shí)例,及實(shí)現(xiàn)效果圖,矢量室內(nèi)對(duì)圖簡(jiǎn)單DEMO,需要的朋友可以參考下2016-11-11
Android設(shè)計(jì)模式之單例模式實(shí)例
這篇文章主要介紹了Android設(shè)計(jì)模式之單例模式實(shí)例,單例模式是運(yùn)用最廣泛的設(shè)計(jì)模式之一,在應(yīng)用這個(gè)模式時(shí),單例模式的類(lèi)必須保證只有一個(gè)實(shí)例存在2023-04-04
Android使用 Coroutine + Retrofit打造簡(jiǎn)單的HTTP請(qǐng)求庫(kù)
這篇文章主要介紹了Android使用 Coroutine + Retrofit打造簡(jiǎn)單的HTTP請(qǐng)求庫(kù),幫助大家更好的理解和學(xué)習(xí)使用Android,感興趣的朋友可以了解下2021-03-03

