xposed 配置如何传入指定模块(ContentProvider)

本文探讨了如何在编写有界面的Xposed模块时,通过ContentProvider在不同进程中安全地传递用户配置数据。介绍了Uri、ContentResolver的使用方法,包括查询、插入、更新和删除操作,以及UriMatcher和MIME类型的理解。重点展示了如何在AndroidManifest中注册ContentProvider和在hook代码中实施数据操作。
摘要由CSDN通过智能技术生成

当我们编写有界面的xposed模块时,我们所给用户提供了可以配置的数据,但是配置的界面是在当前进程,而执行的hook代码在另外一个进程,那么怎么传递这些数据呢?

根据研究有如下三个方法
1.XSharedPreferences(android 7.0以后便不能用了)
2.文件(不确定目标程序是否开启权限,也不适用)
3.contentprovider (可用)

所以最后就决定了使用contentprovider

Uri

首先我们需要了解Uri,Uri由两部分组成 authority和path
authority:用来区分不同的程序,一般以包名命名
path:对同一程序中的表的区分
Uri前部需要添加协议说明
例如:content://com.xxx.xxx/user(com.xxx.xxx中的user表)
如果我们希望查询user表中id为1的数据应该这样写

content://com.xxx.xxx/user/1
*:表示任意长度的字符 content://com.xxx.xxx/*
\#:表示任意长度的数字 content://com.xxx.xxx/user/#

ContentResolver

要想使用contentprovider,就需要借助到contentresolver。通过context.getContentResolver();来获取该类。其实较为常用的方法有四种 insert,updata,delete,query

query:

 Cursor querys = getContentResolver().query(Uri.parse("content://com.xxx.xxx/user"), null, null, null, null);
        if (querys != null) {
            while (querys.moveToNext()) {
                querys.getString(querys.getColumnIndex("packname")));//获取数据中的packname字段的值
            
        }
        querys.close();
query(uri,projection,selection,selectionArgs,sortorder)
uri:获取到的uri
projection:查询的列名
selection:查询的条件 例如“packname=?”
selecionArgs:查询条件的值 例如:new String[]{"123"}
sortotder:排序的方式

insert

 ContentValues contentValues = new ContentValues();
 contentValues.put("packname", pageitens.get(position).packname);
 context.getContentResolver().insert(parse, contentValues);
我的表设计的简单 只有两个字段 一个id(自增) 一个packname 

update

ContentValues contentValues = new ContentValues();
 contentValues.put("packname", pageitens.get(position).packname);
 context.getContentResolver().update(parse, contentValues,“packname=?”,new String[]{"123"});
这个也比较好理解
update(uri,value,selection,selectinArgs)
uri:获取到的uri
value:要修改的数据
selection:查询的条件 例如“packname=?”
selecionArgs:查询条件的值 例如:new String[]{"123"}

delete

 context.getContentResolver().delete(parse, "packname=?", new String[]{pageitens.get(position).packname});
delete(uri,selection,selectinArgs)
uri:获取到的uri
selection:查询的条件 例如“packname=?”
selecionArgs:查询条件的值 例如:new String[]{"123"}

ContentPeovider

public class TestContentProvider extends ContentProvider {


    public static final String AUTHO = "com.xxx.xxx";
    public static final int USER = 1;
    public static final int USERITEM = 2;
    public static final int USERITEM1 = 3;


    private DBHelper dbHelper;
    private UriMatcher uriMatcher;

    {
        //用来筛选uri
        uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
        uriMatcher.addURI(AUTHO, "user", USER);
        //#表示任意数字  *表示任意长度字符
        uriMatcher.addURI(AUTHO, "user/#", USERITEM);
        uriMatcher.addURI(AUTHO, "*", USERITEM1);

    }

    @Override
    public int delete(Uri uri, String selection, String[] selectionArgs) {
        // Implement this to handle requests to delete one or more rows.
        SQLiteDatabase writableDatabase = dbHelper.getWritableDatabase();
        int deleteint = 0;
        switch (uriMatcher.match(uri)) {
            //根据uri筛选表
            case USER:
                //删除数据
                deleteint = writableDatabase.delete("user", selection, selectionArgs);
                break;
        }
        return deleteint;
    }

    @Override
    public String getType(Uri uri) {
        /**
         * MIME两种类型
         * 1.uri以路径结尾 前接vnd.android.cursor.dir/vnd.authority.path
         * 2.uri以id结尾 后接vnd.android.cursor.item/vnd.authority.path
         */
        switch (uriMatcher.match(uri)) {
            case USER:
                return "vnd.android.cursor.dir/vnd.com.xxx.xxx/user";

            case USERITEM:
                return "vnd.android.cursor.item/vnd.com.xxx.xxx/user";

            case USERITEM1:
                return "vnd.android.cursor.item/vnd.com.xxx.xxx/user";

            default:
                break;
        }
        return null;
    }

    @Override
    public Uri insert(Uri uri, ContentValues values) {
        SQLiteDatabase writableDatabase = dbHelper.getWritableDatabase();
        Uri inserturi = null;
        switch (uriMatcher.match(uri)) {
            //根据uri筛选表
            case USER:
                //添加数据
                long user = writableDatabase.insert("user", null, values);
                inserturi = Uri.parse("content://" + AUTHO + "/user/" + user);
                break;
            default:
                throw new IllegalStateException("Unexpected value: " + uriMatcher.match(uri));
                
        }
        return inserturi;
    }

    @Override
    public boolean onCreate() {
        //初始化数据库
        dbHelper = new DBHelper(getContext());
        return true;
    }

    @Override
    public Cursor query(Uri uri, String[] projection, String selection,
                        String[] selectionArgs, String sortOrder) {
        SQLiteDatabase readableDatabase = dbHelper.getReadableDatabase();
        Cursor cursor = null;
        //根据uri筛选表
        switch (uriMatcher.match(uri)) {
            case USER:
                cursor = readableDatabase.query("user", projection, selection, selectionArgs, null, null, sortOrder);
                break;
        }
        return cursor;
    }

    @Override
    public int update(Uri uri, ContentValues values, String selection,
                      String[] selectionArgs) {
        SQLiteDatabase writableDatabase = dbHelper.getWritableDatabase();
        int updatint = 0;
        switch (uriMatcher.match(uri)) {
            case USER:
                updatint = writableDatabase.update("user", values, selection, selectionArgs);
                break;
        }
        return updatint;

    }
}

重点讲解两个方法 UriMatcher,gettype,query(增删改查都差不多)
UriMatcher

UriMatcher主要是用来通过uri来筛选表
1.new UriMatcher(UriMatcher.NO_MATCH);//创建实例
2.uriMatcher.addURI(AUTHO, "user", 参数3);
将uri规则添加进去 AUTHO:authority   “user”:表明  参数3:用于后面区分不同的uri(int类型)
3.switch (uriMatcher.match(uri)) {
            //根据uri筛选表
            case 参数3:
                //删除数据
                deleteint = writableDatabase.delete("user", selection, selectionArgs);
                break;
        }
 通过uriMatcher.match(uri)获取到对于的adduri中的参数3

getType

MIME两种类型
         * 1.uri以路径结尾 前接vnd.android.cursor.dir/vnd.authority.path
         * 2.uri以id结尾 后接vnd.android.cursor.item/vnd.authority.path
         */
         代码中有案例

最后记得在AndroidManifest中对刚刚写的ContentProvider进行注册 就可以了

 <provider
            android:name=".TestContentProvider"
            android:authorities="com.xxx.xxx"
            android:enabled="true"
            android:exported="true" />

在hook代码中通过ContentResolver根据Uri进行数据的操作即可

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值