前言
Xposed开源框架网上有很多的资源,其使用原理和使用方法有很多的资料了,但是为了加深自己对Xpose原理的理解,还是参考大神的自己理解总结了下,其原理通过替换/system/bin/app_process程序来控制Zygote进程,在app_process进程启动时会加载XposeBridge.jar这个jar包,从而完成对Zygote进程和虚拟机的劫持。
在Android系统中,应用程序进程都是由Zygote进程孵化出来的,而Zygote进程是由Init进程启动的。Zygote进程在启动时会创建一个Dalvik虚拟机实例,每当它孵化一个新的应用程序进程时,都会将这个Dalvik虚拟机实例复制到新的应用程序进程里面去,从而使得每一个应用程序进程都有一个独立的Dalvik虚拟机实例。这也是Xposed选择替换app_process的原因。
Zygote进程在启动的过程中,除了会创建一个Dalvik虚拟机实例之外,还会将Java运行时库加载到进程中来,以及注册一些Android核心类的JNI方法来前面创建的Dalvik虚拟机实例中去。注意,一个应用程序进程被Zygote进程孵化出来的时候,不仅会获得Zygote进程中的Dalvik虚拟机实例拷贝,还会与Zygote一起共享Java运行时库。这也就是可以将XposedBridge这个jar包加载到每一个Android应用程序中的原因。XposedBridge有一个私有的Native(JNI)方法hookMethodNative,这个方法也在app_process中使用。
XsharePreference的使用
相信很多人都使用过Xposed框架去hook某个应用编写出很多模块,在编写hook模块的时候往往我们都涉及到数据保存,比如hook某个应用方法的开关控制,我们需要将开关的状态保存,当Xposed执行hook到这个方法是会读取开关的状态,对于保存数据而言有很多种方式,如读写文件、创建数据库保存,使用SharePreference保存等等。我今天所介绍的是Xposed中封装的XSharePreference的Api的使用。
使用方法
方式一 使用 XSharedPreferences(String packageName)获取
我们知道SharePrefernce实际上是对应在/data/data/<Package Name>/shared_prefs/<Package Name>_preferences.xml文件。里面按键值对的形式存放。XSharedPreferences做的工作实际上就是根据<Package Name>自己去解析这个文件,然后读出数据存放在一个HashMap当中。
构造方法源码
public XSharedPreferences(String packageName) {
this(packageName, packageName + "_preferences");
}
通过构造方法可以看出我们在设置SharePrefernece的第一个参数时后面还需要加上"_preference"。
保存数据
SharedPreferences mSharedPreferences = getActivity().getSharedPreferences(getPackageName() + "_preferences", Activity.MODE_WORLD_READABLE);
SharedPreferences.Editor mEditor = mSharedPreferences.edit();
mEditor.putBoolean(PreferenceUtils.AUTO_ALL, (Boolean) newValue).commit();
mEditor.putString(PreferenceUtils.REPLY_CONTENT,(String)newValue).commit();
注意在设置SharePrefernce的模式是设置成
Activity.MODE_WORLD_READABLE,只有这样XSharePreference才能访问。但是当Xposed中使用XSharepreference在7.0系统上时会遇到安全异常,XsharePreference会失效,此时最好的解决方法是使用ContentProvider代替使用。
数据访问
XSharedPreferences intance = new XSharedPreferences("com.ydscience.fakemomo");
方式二 使用public XSharedPreferences(String packageName, String prefFileName) (推荐使用)
XSharePreference的第二种初始化构造方法源码为
public XSharedPreferences(String packageName, String prefFileName) {
mFile = new File(Environment.getDataDirectory(), "data/" + packageName + "/shared_prefs/" + prefFileName + ".xml");
mFilename = mFile.getAbsolutePath();
startLoadFromDisk();
}
对比两个构造方法而言,第一个初始化完成后继续调用第二个初始化构造函数,但是对比而言个人还是喜欢第二种,使用简单清晰。
保存数据
SharedPreferences mSharedPreferences = getActivity().getSharedPreferences("config", Activity.MODE_WORLD_READABLE);
SharedPreferences.Editor mEditor = mSharedPreferences.edit();
mEditor.putBoolean(PreferenceUtils.AUTO_ALL, (Boolean) newValue).commit(); mEditor.putString(PreferenceUtils.REPLY_CONTENT,(String)newValue).commit();
数据访问
在数据访问时封装了一个类PreferenceUtils类,完整代码为
public class PreferenceUtils {
public static final String AUTO_ALL ="auto_all";
public static final String REPLY_CONTENT ="reply_content";
public static final String REPLY_ORDER ="reply_order";
public static final String REPLY_ROBOT ="reply_robot";
public static final String REPLY_DESCRIBE ="reply_describe";
private static XSharedPreferences intance = null;
public static XSharedPreferences getIntance(){
if (intance == null){
intance = new XSharedPreferences("com.ydscience.fakemomo","config");
intance.makeWorldReadable();
}else {
intance.reload();
}
return intance;
}
public static boolean openAll(){
return getIntance().getBoolean(AUTO_ALL,false);
}
public static String replyContent(){
return getIntance().getString(REPLY_CONTENT,"请设置自动回复内容");
}
}