当我们编写有界面的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进行数据的操作即可