csonezp

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::

看了一下文档,发现Xposed这货实在太强大了,它直接对Zygote做了一些手脚,简直是无所不能。缺点就是需要安装一下框架。基本原理就是把Android 系统的app_process给替换了,然后启动Zygote的时候多加载点东西,通过一些特殊手段直接把Zygote和Dalvki给劫持了。本文就将带领读者去完成一个简单的Xposed的Module,让大家领略一下Xposed的威力。

要想能看到程序运行的结果,你首先要有一部Root过,安装过Xposed框架和XposedInstaller的手机。这部分不在本篇教程范围内,请自行百度。Xposed的官方下载地址

项目的源码地址:click me

一,建立项目

AS工程,本人AS版本2.0 p8,这个版本建立完工程需要做一个特殊操作,要把Project的build文件里的

 classpath 'com.android.tools.build:gradle:2.0.0-alpha8'

换成

 classpath 'com.android.tools.build:gradle:1.5.0'

不然无法开发Xposed Module。

 

然后需要在这里 下载XposedBridgeApi-XX.jar,XX为最新版本号。这个是Xposed Module开发需要的Lib。

将XposedBridgeApi-XX.jar引入项目。

 

 

 

二,Android App部分开发。

这部分就很简单了,是一个EditText,一个按钮,在EditText中输入数字3然后点击按钮后会显示yes,否则就是no。

AS建立项目后默认会有一个名为app的module(这个是安卓项目的module),在这个module下开发就行,不需要另外开modu。

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    >

    <EditText
        android:id="@+id/edit"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:hint="请输入3" />

    <Button
        android:id="@+id/btn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_below="@id/edit"
        android:layout_centerHorizontal="true"
        android:text="测试" />

</RelativeLayout>
public class MainActivity extends AppCompatActivity {
    Button btn;
    EditText edit;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        btn = (Button) findViewById(R.id.btn);
        edit = (EditText) findViewById(R.id.edit);
        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                int a;
                try {
                    a = Integer.parseInt(edit.getText().toString());
                } catch (Exception e) {
                    Toast.makeText(MainActivity.this, "Oh!No!!", Toast.LENGTH_SHORT).show();
                    return;
                }
                if (isEquals(a)) {
                    Toast.makeText(MainActivity.this, "Yes!you got it!!", Toast.LENGTH_SHORT).show();
                } else {
                    Toast.makeText(MainActivity.this, "Oh!No!!", Toast.LENGTH_SHORT).show();
                }
            }
        });
    }

    private boolean isEquals(int a) {
        if (a == 3) {
            return true;
        }
        return false;
    }
}

很简单的一个安卓程序,不需要过多解释了。

 

三,Xposed部分开发

这个项目的xposed module的功能是hook本项目Android 程序部分,让 Activity的isEquls(int a)方法始终返回true,这样无论我们输入什么数字,最终都会显示yes。

修改清单文件:

<meta-data
            android:name="xposedmodule"
            android:value="true" />
        <meta-data
            android:name="xposeddescription"
            android:value="Test Xposed" />
        <meta-data
            android:name="xposedminversion"
            android:value="46" />

这些是声明这个项目是一个xposed module。其中xposedminversion表示支持的最低的xposed api版本。

 

建立一个Java Class文件:

/**
 * Created by csonezp on 16-2-2.
 */
public class HookDemo implements IXposedHookLoadPackage {
    @Override
    public void handleLoadPackage(XC_LoadPackage.LoadPackageParam loadPackageParam) throws Throwable {
        //你想操作的应用的包名,这里为这个项目自身的包名
        if (loadPackageParam.packageName.equals("com.csonezp.xposeddemo2")) {
            Class clasz = loadPackageParam.classLoader.loadClass("com.csonezp.xposeddemo2.MainActivity");
            //hook MainActivity的 isEquls 方法
            XposedHelpers.findAndHookMethod(clasz, "isEquals", int.class, new XC_MethodHook() {
                //这俩个方法任选其一就可以达到效果
                //方法执行前进行的操作
                @Override
                protected void beforeHookedMethod(MethodHookParam param) throws Throwable {
                    Integer para1 = (Integer) param.args[0];   //获取参数1
                    String s1 = Integer.toString(para1);
                    Log.v("hook before param1:", s1);
                    param.args[0] = 3;  //设置参数1,也就是将isEquls的参数恒定设为3
                    Log.v("hook", "before hook!");
                    XposedBridge.log("hook before param1:" + s1);
                }

                @Override
                protected void afterHookedMethod(MethodHookParam param) throws Throwable {
                    boolean res = (boolean) param.getResult();
                    Log.v("hook after result :", res + "");
                    Integer para1 = (Integer) param.args[0];   //获取参数1
                    String s1 = Integer.toString(para1);
                    param.setResult(true);   //设置返回值,始终为true
                    XposedBridge.log("hook after result:true");
                    Log.v("hook param1:", s1);
                }
            });
        }
    }
}

这就是这个Xposed Module的核心方法了。这个类实现了IXposedHookLoadPackage接口,当读取系统Package的时候,就会调用handleLoadPackage方法。

这里handleLoadPackage主要对“com.csonezp.xposeddemo2” 这个包(也就是本项目)的MainActivity 的 isEquals方法进行一些操作,让这个方法始终返回true。

 

然后就要添加一个assets文件夹。

右键点击 app这个module  --> new --> Folder -->Assets Folder,然后确认即可。

在assets文件夹中建立一个xposed_init文件

文件内容就是我们上面写的那个Java Class的类名,Xposed框架就是通过这里加载我们的类的,所以一定不要写错。

 

OK,大功告成,将程序装到手机上,在XposedInstaller中开启我们的Module,重启手机,再打开App,你会发现无论在EditText中输入什么数字,显示的都是Yes了

posted on 2016-02-02 16:39  csonezp  阅读(11087)  评论(6编辑  收藏  举报