dexposed框架Android在线热修复
移动client应用相对于Webapp的最大一个问题每次出现bug,不能像web一样在server就完毕修复,不须要发版本号。紧急或者有安全漏洞的问题,
假设是Webapp你可能最多花个1,2个小时紧急公布上线。可是app呢,打包,跪求市场公布几百个渠道,周末还发不了。app配置升级。你还不能配置
强制升级。 就算配置提示升级,用户心里肯定想前两天刚升级最新版,怎么又要升。并且升级要流量。这时候会非常反感甚至卸载应用。所以安卓是否
有能力做到在线打补丁?dexposed给我们攻克了这个问题。
1、淘宝Dexposed框架
喜欢刷机的人应该对xposed不陌生,Xposed框架是一款能够在不改动APK的情况下影响程序执行(改动系统)的框架服务。基于它 能够制作出很多功
能强大的模块。
事实上Xposed安卓的一个开源框架,在github上的下载地址xposed,有兴趣的能够去研究下,dexPosed也是基于Xposed的。
他有几个典型的使用场景:
a. Classic AOP programming (aop编程)
b. Instrumentation (for testing, performance monitoring and etc.) 測试,性能监控
c. Online hot patch to fix critical, emergent or security bugs 线上打补丁,解决一些严重的,紧急的或者安全漏洞的bug。
d. SDK hooking for a better development experience
对于aop编程我这里就不多说了,deXposed提供的 是无侵入性的,并且AOP就是用的java代码。相比国外比較流行的Aspectj有几点优势:
a、无侵入性。
b、使用Java代码编写,Aspectj使用脚本须要一定的学习成本
c、Aspectj有自己的编译器,须要编译下Aspectj代码。会注入他自己的代码
以下来看看假设写一些AOP的代码:
Attach a piece of code before and after all occurrences of Activity.onCreate(Bundle).
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">
beforeHookedMethod和afterHookedMethod方法做一些详细的操作。我们能够看到,用dexposed能够实现不改变原函
数的运行。可是在原函数运行前后去做一些其它的额外处理。比如改变入參和返回值等等的一些事情。
Replace the original body of the target method. DexposedBridge.findAndHookMethod(Activity.class, "onCreate", Bundle.class, new XC_MethodReplacement() { @Override protected Object replaceHookedMethod(MethodHookParam param) throws Throwable {
// Re-writing the method logic outside the original method context is a bit tricky but still viable.
...
} });
则是能够将原有要运行的函数替换成一个我们须要的新的运行函数。
这个框架眼下对android 5.0系统支持度不高。只是框架更新的时间能够看出,持续维护中,不久将对art全然支持。
他提供一个函数来检測你的android
系统是否支持 Dexposed,DexposedBridge.canDexposed(context)。
你须要把补丁包打包成一个apk。 然后通过以下的代码来载入补丁包:
// Run taobao.patch apk
public void runPatchApk() {
if (android.os.Build.VERSION.SDK_INT == 21) {
return;
}
if (!isSupport) {
Log.d("dexposed", "This device doesn't support dexposed!");
return;
}
File cacheDir = getExternalCacheDir();
if (cacheDir != null) {
String fullpath = cacheDir.getAbsolutePath() + File.separator + "patch.apk";
PatchResult result = PatchMain.load(this, fullpath, null);
if (result.isSuccess()) {
Log.e("Hotpatch", "patch success!");
} else {
Log.e("Hotpatch", "patch error is " + result.getErrorInfo());
}
}
}
2.Dexposed原理
重点是搞清楚是如何hook的。
1)首先通过ClassLoader把插件apk载入进入主project
2)通过反射拿到详细的class类
3)DexposedBridge.findAndHookMethod定位到详细的方法中,对代码进行覆盖或者增加自己定义功能
前面两点都是Java方法没什么能够说的,我这里重点分析下第三点:
a、DexposedBridge.findAndHookMethod()中调用了Xc_methdhook.hookMethos(),这种方法是通过传入的class。方法及签名去查找返回一个相应的Method。
然后把Method作为參数传入hookMethodNative(Method ...);
b、hookMethodNative是进行hook的主要方法。分析源代码发现,它里面做了这些事情
1、把Java层的变量,类型转换成c能识别的指针类型。
// Save a copy of the original method and other hook info
DexposedHookInfo* hookInfo = (DexposedHookInfo*) calloc(1, sizeof(DexposedHookInfo));
memcpy(hookInfo, method, sizeof(hookInfo->originalMethodStruct));
hookInfo->reflectedMethod = dvmDecodeIndirectRef(dvmThreadSelf(), env->NewGlobalRef(reflectedMethodIndirect));
hookInfo->additionalInfo = dvmDecodeIndirectRef(dvmThreadSelf(), env->NewGlobalRef(additionalInfoIndirect));
2、用我们自己的code取代,看到以下的nativeFun了吗。这种方法被dexposedCallHandler取代。也就是说被我们自己的回调方法取代了。这样我们就能够
任意的操作了。然后把要hook的对象參数保存到insns
// Replace method with our own code
SET_METHOD_FLAG(method, ACC_NATIVE);
method->nativeFunc = &dexposedCallHandler;
method->insns = (const u2*) hookInfo;
method->registersSize = method->insSize;
method->outsSize = 0;
3、接下来我们去看看dexposedCallHandler做了哪些事情;
//赋值
DexposedHookInfo* hookInfo = (DexposedHookInfo*) method->insns;
Method* original = (Method*) hookInfo;
Object* originalReflected = hookInfo->reflectedMethod;
Object* additionalInfo = hookInfo->additionalInfo;
// call the Java handler function
//对...这才是重点。dvmCallMethod方法回调了dexposedHandleHookedMethod方法,这种方法是在Java层实现的
JValue result;
dvmCallMethod(self, dexposedHandleHookedMethod, NULL, &result,
originalReflected, (int) original, additionalInfo, thisObject, argsArray); dvmReleaseTrackedAlloc((Object *)argsArray, self);
4、我们在去看看handleHookMethod做了什么。一看你就明朗了;
private static Object handleHookedMethod(Member method, int originalMethodId, Object additionalInfoObj,
Object thisObject, Object[] args) throws Throwable {
AdditionalHookInfo additionalInfo = (AdditionalHookInfo) additionalInfoObj; if (disableHooks) {
try {
return invokeOriginalMethodNative(method, originalMethodId, additionalInfo.parameterTypes,
additionalInfo.returnType, thisObject, args);
} catch (InvocationTargetException e) {
throw e.getCause();
}
} Object[] callbacksSnapshot = additionalInfo.callbacks.getSnapshot();
final int callbacksLength = callbacksSnapshot.length;
if (callbacksLength == 0) {
try {
return invokeOriginalMethodNative(method, originalMethodId, additionalInfo.parameterTypes,
additionalInfo.returnType, thisObject, args);
} catch (InvocationTargetException e) {
throw e.getCause();
}
} MethodHookParam param = new MethodHookParam();
param.method = method;
param.thisObject = thisObject;
param.args = args; // call "before method" callbacks
int beforeIdx = 0;
do {
try {
((XC_MethodHook) callbacksSnapshot[beforeIdx]).beforeHookedMethod(param);
} catch (Throwable t) {
DexposedBridge.log(t); // reset result (ignoring what the unexpectedly exiting callback did)
param.setResult(null);
param.returnEarly = false;
continue;
} if (param.returnEarly) {
// skip remaining "before" callbacks and corresponding "after" callbacks
beforeIdx++;
break;
}
} while (++beforeIdx < callbacksLength); // call original method if not requested otherwise
if (!param.returnEarly) {
try {
param.setResult(invokeOriginalMethodNative(method, originalMethodId,
additionalInfo.parameterTypes, additionalInfo.returnType, param.thisObject, param.args));
} catch (InvocationTargetException e) {
param.setThrowable(e.getCause());
}
} // call "after method" callbacks
int afterIdx = beforeIdx - 1;
do {
Object lastResult = param.getResult();
Throwable lastThrowable = param.getThrowable(); try {
((XC_MethodHook) callbacksSnapshot[afterIdx]).afterHookedMethod(param);
} catch (Throwable t) {
DexposedBridge.log(t); // reset to last result (ignoring what the unexpectedly exiting callback did)
if (lastThrowable == null)
param.setResult(lastResult);
else
param.setThrowable(lastThrowable);
}
} while (--afterIdx >= 0); // return
if (param.hasThrowable())
throw param.getThrowable();
else
return param.getResult();
}
推断是否hook成功。假设不成功运行invokeOriginalMethodNative,也就是运行原始函数;
成功则,这个函数查找被挂钩函数的挂钩 XC_MethodHook结构体,然后运行里边的 beforeHookedMethod函数。再通过 invokeOriginalMethodNative
运行挂钩前的原始函数。最后再运行 afterHookedMethod 函数。
findAndHookMethod 的实现就分析完了,
本质上仍然是寻找被挂钩函数的 Method 结构体,将Method属性改为native ,然后对其成员 nativeFunc,
registersize 等进行赋值,当中 insns 成员保存了挂钩的具体信息。全部被挂钩的函数。其nativeFunc都赋值为 dexposedCallHandler 函数,该函数终于运行 XposedBridge.class 里的 handleHookedMethod 。 handleHookedMethod 寻找dexposed模块及dexposed框架调用 findAndHookMethod 注冊的 before,after
函数,假设有,就运行,再通过invokeOriginalMethodNative 运行挂钩前函数。
3、怎么使用dexposed
a、下载dexposed源代码 dexposed
b、把下载的源代码导入,仅仅须要sample包。当中dexposedsample是有bug的主project,patchsample为插件project,(其它的两个为工具类源代码,方便我们理解源代码)
c、主project配置gradle文件
apply plugin: 'com.android.application'
android {
compileSdkVersion 21
buildToolsVersion "21.1.2" defaultConfig {
applicationId "com.taobao.dexposed"
minSdkVersion 9
targetSdkVersion 21
versionName "1.0"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
} android {
packagingOptions {
exclude 'AndroidManifest.xml'
} } buildTypes {
//Debug打开,使用測试环境接口
debug {
}
}
lintOptions {
abortOnError false
} sourceSets {
main {
manifest.srcFile 'src/main/AndroidManifest.xml'
java.srcDirs = ['src/main/java']
res.srcDirs = ['src/main/res'] //路径依据自己的相应改动
}
}
} //我使用的是本地导入so包。
task copyNativeLibs(type: Copy) {
from fileTree(dir: 'libs', include: '*/*.so' ) into 'build/native-libs'
}
tasks.withType(JavaCompile) { compileTask -> compileTask.dependsOn copyNativeLibs } clean.dependsOn 'cleanCopyNativeLibs' tasks.withType(com.android.build.gradle.tasks.PackageApplication) {
pkgTask ->
pkgTask.jniFolders = new HashSet<File>()
pkgTask.jniFolders.add(new File(projectDir, 'libs'))
}
//---------------------------------
//这段配置是把主project打成jar包。这个jar包须要导入到patchproject中
task clearJar(type: Delete) {
delete 'libs/sdk.jar'
} task makeJar(type:org.gradle.api.tasks.bundling.Jar) {
//指定生成的jar名
baseName 'sdk'
//从哪里打包class文件
from('build/intermediates/classes/sample/debug/com')
//打包到jar后的文件夹结构,依据自己project须要写路径
into('com')
//去掉不须要打包的文件夹和文件
exclude('test/', 'BuildConfig.class', 'R.class')
//去掉R$开头的文件
exclude{ it.name.startsWith('R$');}
} makeJar.dependsOn(clearJar, build)
//--------------------------------------------- dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
compile files('libs/dexposedbridge.jar')
}
patchprojectgradle配置:
dependencies {
// compile fileTree(dir: 'libs', include: ['*.jar'])
provided files('libs/sdk.jar')
provided files('libs/dexposedbridge.jar')
provided files('libs/patchloader.jar')
}
使用provided方式导入jar包,这样仅仅会在编译时引用jar,不会把jar打包进patch.apk。防止与主project引入包冲突;
用例代码就不贴了,自己去down一份
4、关于patch包的安全问题能够看Android Hotpatch
dexposed框架Android在线热修复的更多相关文章
- Android 模块化/热修复/插件化 框架选用
概念汇总 动态加载:在程序运行的时候,加载一些程序自身原本不存在的文件并运行这些文件里的代码逻辑.动态加载是热修复与插件化实现的基础. 热修复:修改部分代码,不用重新发包,在用户不知情的情况下,给ap ...
- 包建强的培训课程(15):Android App热修复技术
@import url(/css/cuteeditor.css); Normal 0 10 pt 0 2 false false false EN-US ZH-CN X-NONE $([{£¥·‘“〈 ...
- Android RocooFix热修复动态加载框架介绍
RocooFix Another hotfix framework 之前的HotFix项目太过简单,也有很多同学用Nuwa遇到很多问题,作者也不再修复,所以重新构建了一套工具. Bugfix 2016 ...
- 手把手带你打造一个 Android 热修复框架
本文来自网易云社区 作者:王晨彦 Application 处理 上面我们已经对所有 class 文件插入了 Hack 的引用,而插入 dex 是在 Application 中,Application ...
- Android 热补丁和热修复
参考: 各大热补丁方案分析和比较 Android App 线上热修复方案 1. Xposed Github地址:https://github.com/rovo89/Xposed 项目描述:Xposed ...
- 十分钟教会你使用安卓热修复框架AndFix
腾讯最近开发出一个Tinker,阿里也有一个Dexposed框架,当然还有一个就是今天的主角热修复框架AndFix.接下来,我会从它的概念.原理.使用方法等为你详细介绍. 1.什么是AndFix? A ...
- Android热修复技术原理详解(最新最全版本)
本文框架 什么是热修复? 热修复框架分类 技术原理及特点 Tinker框架解析 各框架对比图 总结 通过阅读本文,你会对热修复技术有更深的认知,本文会列出各类框架的优缺点以及技术原理,文章末尾简单 ...
- 热修复干货| AndFix热补丁动态修复框架使用教程
本篇文章会与大家一起学习使用阿里的AndFix热修复框架,可以说AndFix是国内热修复技术的开山始祖,尽管现在阿里已经放弃了对这个项目的维护,但是后来很多的热修复技术都借鉴了这一框架的实现思路. 1 ...
- 探索安卓热修复框架AndFix的奥秘
虽然阿里的AndFix框架已经出来很长时间了,但是还不了解它的同学依然挺多,接下来就跟着我一起来到AndFix的世界里一起看看,如何达到不用重新安装app就可以修复bug. 1.什么是AndFix? ...
随机推荐
- java 微信server录音下载到自己server
/** * @author why * */ public class VoiceDownload { /** * * 依据文件id下载文件 * * * * @param mediaId * * 媒体 ...
- 使用roslyn编译website项目
在Nuget中,添加Microsoft.CodeDom.Providers.DotNetCompilerPlatform. 在添加这个dll的时候,会自动在web.config中添加以下内容 < ...
- [SCOI 2008] 奖励关
[题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=1076 [算法] f[i][S]表示当前第i次抛出宝物,目前集合为S,所能获得的最高分 ...
- docker compose线下安装
Compose 是一个用户定义和运行多个容器的 Docker 应用程序.在 Compose 中你可以使用 YAML 文件来配置你的应用服务.然后,只需要一个简单的命令,就可以创建并启动你配置的所有服务 ...
- 几种AutoLayout自动布局所经常使用的布局约束类型
width表示约束ui控件的固定宽度 height表示约束ui控件的固定高度 Leading Space to Superview 与父视图的左边界线保持固定距离 Trailing Space to ...
- centOS 7安装mysql5.6
方法二:官网下载安装mysql-server # wget http://dev.mysql.com/get/mysql-community-release-el7-5.noarch.rpm # rp ...
- Spark RDD概念学习系列之Pair RDD的分区控制
不多说,直接上干货! Pair RDD的分区控制 Pair RDD的分区控制 (1) Spark 中所有的键值对RDD 都可以进行分区控制---自定义分区 (2)自定义分区的好处: 1) 避免数据倾 ...
- 【转】.NET MVC控制器分离到类库的方法
在.ASP.NET MVC的开发中,我们创建完项目之后,ASP.NET MVC是已Model-Controller-View的形式存在的,在创建项目自动生成的内容上Model我们很容易分离成类库,所以 ...
- 基于任务的编程模型TAP
一.引言 在上两个专题中我为大家介绍.NET 1.0中的APM和.NET 2.0中的EAP,在使用前面两种模式进行异步编程的时候,大家多多少少肯定会感觉到实现起来比较麻烦, 首先我个人觉得,当使用AP ...
- dragView 屏幕拖拽并且弹出菜单的控件
dragView 因项目新需求需要添加一个屏幕拖拽按钮可以弹出菜单的控件,因为不是我做的闲来无事写一个demo吧 可能存在一些小bug(毕竟就写了几个小时)兄弟姐妹们理解思路就行 具体的可以自己调试一 ...