- 作用:操作
URI
- 具体使用
核心方法有两个:withAppendedId()
&parseId()
// withAppendedId()作用:向URI追加一个id
Uri uri = Uri.parse(“content://cn.scu.myprovider/user”)
Uri resultUri = ContentUris.withAppendedId(uri, 7);
// 最终生成后的Uri为:content://cn.scu.myprovider/user/7
// parseId()作用:从URL中获取ID
Uri uri = Uri.parse(“content://cn.scu.myprovider/user/7”)
long personid = ContentUris.parseId(uri);
//获取的结果为:7
4.6 UriMatcher类
- 作用
- 在
ContentProvider
中注册URI
- 根据
URI
匹配ContentProvider
中对应的数据表
- 具体使用
// 步骤1:初始化UriMatcher对象
UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH);
//常量UriMatcher.NO_MATCH = 不匹配任何路径的返回码
// 即初始化时不匹配任何东西
// 步骤2:在ContentProvider 中注册URI(addURI())
int URI_CODE_a = 1;
int URI_CODE_b = 2;
matcher.addURI(“cn.scu.myprovider”, “user1”, URI_CODE_a);
matcher.addURI(“cn.scu.myprovider”, “user2”, URI_CODE_b);
// 若URI资源路径 = content://cn.scu.myprovider/user1 ,则返回注册码URI_CODE_a
// 若URI资源路径 = content://cn.scu.myprovider/user2 ,则返回注册码URI_CODE_b
// 步骤3:根据URI 匹配 URI_CODE,从而匹配ContentProvider中相应的资源(match())
@Override
public String getType (Uri uri){
Uri uri = Uri.parse(" content://cn.scu.myprovider/user1");
switch (matcher.match(uri)) {
// 根据URI匹配的返回码是URI_CODE_a
// 即matcher.match(uri) == URI_CODE_a
case URI_CODE_a:
return tableNameUser1;
// 如果根据URI匹配的返回码是URI_CODE_a,则返回ContentProvider中的名为tableNameUser1的表
case URI_CODE_b:
return tableNameUser2;
// 如果根据URI匹配的返回码是URI_CODE_b,则返回ContentProvider中的名为tableNameUser2的表
}
}
4.7 ContentObserver类
-
定义:内容观察者
-
作用:观察 Uri引起ContentProvider 中的数据变化 & 通知外界(即访问该数据访问者)
当
ContentProvider
中的数据发生变化(增、删 & 改)时,就会触发该ContentObserver
类
- 具体使用
// 步骤1:注册内容观察者ContentObserver
getContentResolver().registerContentObserver(uri);
// 通过ContentResolver类进行注册,并指定需要观察的URI
// 步骤2:当该URI的ContentProvider数据发生变化时,通知外界(即访问该ContentProvider数据的访问者)
public class UserContentProvider extends ContentProvider {
public Uri insert(Uri uri, ContentValues values) {
db.insert(“user”, “userid”, values);
getContext().getContentResolver().notifyChange(uri, null);
// 通知访问者
}
}
// 步骤3:解除观察者
getContentResolver().unregisterContentObserver(uri);
// 同样需要通过ContentResolver类进行解除
至此,关于ContentProvider
的使用已经讲解完毕
五、 实例说明
-
由于
ContentProvider
不仅常用于进程间通信,同时也适用于进程内通信 -
所以本实例会采用ContentProvider讲解:
- 进程内通信
- 进程间通信
- 实例说明:采用的数据源是
Android
中的SQLite
数据库
5.1 进程内通信
- 步骤说明:
- 创建数据库类
- 自定义
ContentProvider
类 - 注册 创建的
ContentProvider
类 - 进程内访问
ContentProvider
的数据
- 具体使用
步骤1:创建数据库类
DBHelper.java
public class DBHelper extends SQLiteOpenHelper {
// 数据库名
private static final String DATABASE_NAME = “finch.db”;
// 表名
public static final String USER_TABLE_NAME = “user”;
public static final String JOB_TABLE_NAME = “job”;
private static final int DATABASE_VERSION = 1;
//数据库版本号
public DBHelper(Context context) {
super(context, DATABASE_NAME, null, DATABASE_VERSION);
}
@Override
public void onCreate(SQLiteDatabase db) {
// 创建两个表格:用户表 和职业表
db.execSQL(“CREATE TABLE IF NOT EXISTS " + USER_TABLE_NAME + “(_id INTEGER PRIMARY KEY AUTOINCREMENT,” + " name TEXT)”);
db.execSQL(“CREATE TABLE IF NOT EXISTS " + JOB_TABLE_NAME + “(_id INTEGER PRIMARY KEY AUTOINCREMENT,” + " job TEXT)”);
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
}
}
步骤2:自定义 ContentProvider 类
public class MyProvider extends ContentProvider {
private Context mContext;
DBHelper mDbHelper = null;
SQLiteDatabase db = null;
public static final String AUTOHORITY = “cn.scu.myprovider”;
// 设置ContentProvider的唯一标识
public static final int User_Code = 1;
public static final int Job_Code = 2;
// UriMatcher类使用:在ContentProvider 中注册URI
private static final UriMatcher mMatcher;
static {
mMatcher = new UriMatcher(UriMatcher.NO_MATCH);
// 初始化
mMatcher.addURI(AUTOHORITY, “user”, User_Code);
mMatcher.addURI(AUTOHORITY, “job”, Job_Code);
// 若URI资源路径 = content://cn.scu.myprovider/user ,则返回注册码User_Code
// 若URI资源路径 = content://cn.scu.myprovider/job ,则返回注册码Job_Code
}
// 以下是ContentProvider的6个方法
/**
- 初始化ContentProvider
*/
@Override
public boolean onCreate() {
mContext = getContext();
// 在ContentProvider创建时对数据库进行初始化
// 运行在主线程,故不能做耗时操作,此处仅作展示
mDbHelper = new DBHelper(getContext());
db = mDbHelper.getWritableDatabase();
// 初始化两个表的数据(先清空两个表,再各加入一个记录)
db.execSQL(“delete from user”);
db.execSQL(“insert into user values(1,‘Carson’);”);
db.execSQL(“insert into user values(2,‘Kobe’);”);
db.execSQL(“delete from job”);
db.execSQL(“insert into job values(1,‘Android’);”);
db.execSQL(“insert into job values(2,‘iOS’);”);
return true;
}
/**
- 添加数据
*/
@Override
public Uri insert(Uri uri, ContentValues values) {
// 根据URI匹配 URI_CODE,从而匹配ContentProvider中相应的表名
// 该方法在最下面
String table = getTableName(uri);
// 向该表添加数据
db.insert(table, null, values);
// 当该URI的ContentProvider数据发生变化时,通知外界(即访问该ContentProvider数据的访问者)
mContext.getContentResolver().notifyChange(uri, null);
// // 通过ContentUris类从URL中获取ID
// long personid = ContentUris.parseId(uri);
// System.out.println(personid);
return uri;
}
/**
- 查询数据
*/
@Override
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder) {
// 根据URI匹配 URI_CODE,从而匹配ContentProvider中相应的表名
// 该方法在最下面
String table = getTableName(uri);
// // 通过ContentUris类从URL中获取ID
// long personid = ContentUris.parseId(uri);
// System.out.println(personid);
// 查询数据
return db.query(table, projection, selection, selectionArgs, null, null, sortOrder, null);
}
/**
- 更新数据
*/
@Override
public int update(Uri uri, ContentValues values, String selection,
String[] selectionArgs) {
// 由于不展示,此处不作展开
return 0;
}
/**
- 删除数据
*/
@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
// 由于不展示,此处不作展开
return 0;
}
@Override
public String getType(Uri uri) {
// 由于不展示,此处不作展开
return null;
}
/**
- 根据URI匹配 URI_CODE,从而匹配ContentProvider中相应的表名
*/
private String getTableName(Uri uri) {
String tableName = null;
switch (mMatcher.match(uri)) {
case User_Code:
tableName = DBHelper.USER_TABLE_NAME;
break;
case Job_Code:
tableName = DBHelper.JOB_TABLE_NAME;
break;
}
return tableName;
}
}
步骤3:注册 创建的 ContentProvider类
AndroidManifest.xml
步骤4:进程内访问 ContentProvider中的数据
MainActivity.java
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
/**
- 对user表进行操作
*/
// 设置URI
Uri uri_user = Uri.parse(“content://cn.scu.myprovider/user”);
// 插入表中数据
ContentValues values = new ContentValues();
values.put(“_id”, 3);
values.put(“name”, “Iverson”);
// 获取ContentResolver
ContentResolver resolver = getContentResolver();
// 通过ContentResolver 根据URI 向ContentProvider中插入数据
resolver.insert(uri_user,values);
// 通过ContentResolver 向ContentProvider中查询数据
Cursor cursor = resolver.query(uri_user, new String[]{“_id”,“name”}, null, null, null);
while (cursor.moveToNext()){
System.out.println(“query book:” + cursor.getInt(0) +" "+ cursor.getString(1));
// 将表中数据全部输出
}
cursor.close();
// 关闭游标
/**
- 对job表进行操作
*/
// 和上述类似,只是URI需要更改,从而匹配不同的URI CODE,从而找到不同的数据资源
Uri uri_job = Uri.parse(“content://cn.scu.myprovider/job”);
// 插入表中数据
ContentValues values2 = new ContentValues();
values2.put(“_id”, 3);
values2.put(“job”, “NBA Player”);
// 获取ContentResolver
ContentResolver resolver2 = getContentResolver();
// 通过ContentResolver 根据URI 向ContentProvider中插入数据
resolver2.insert(uri_job,values2);
// 通过ContentResolver 向ContentProvider中查询数据
Cursor cursor2 = resolver2.query(uri_job, new String[]{“_id”,“job”}, null, null, null);
while (cursor2.moveToNext()){
System.out.println(“query job:” + cursor2.getInt(0) +" "+ cursor2.getString(1));
// 将表中数据全部输出
}
cursor2.close();
// 关闭游标
}
}
结果
5.2 进程间进行数据共享
- 实例说明:本文需要创建2个进程,即创建两个工程,作用如下
进程1
使用步骤如下:
- 创建数据库类
- 自定义
ContentProvider
类 - 注册 创建的
ContentProvider
类
前2个步骤同上例相同,此处不作过多描述,此处主要讲解步骤3.
步骤3:注册 创建的 ContentProvider类
AndroidManifest.xml
<provider
android:name=“MyProvider”
android:authorities=“scut.carson_ho.myprovider”
// 声明外界进程可访问该Provider的权限(读 & 写)
android:permission=“scut.carson_ho.PROVIDER”
// 权限可细分为读 & 写的权限
// 外界需要声明同样的读 & 写的权限才可进行相应操作,否则会报错
// android:readPermisson = “scut.carson_ho.Read”
// android:writePermisson = “scut.carson_ho.Write”
// 设置此provider是否可以被其他进程使用
android:exported=“true”
/>
// 声明本应用 可允许通信的权限
// 细分读 & 写权限如下,但本Demo直接采用全权限
//
//
至此,进程1创建完毕,即创建ContentProvider
& 数据 准备好了。
进程2
步骤1:声明可访问的权限
AndroidManifest.xml
// 声明本应用可允许通信的权限(全权限)
// 细分读 & 写权限如下,但本Demo直接采用全权限
//
//
// 注:声明的权限必须与进程1中设置的权限对应
步骤2:访问 ContentProvider的类
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
/**
- 对user表进行操作
*/
// 设置URI
Uri uri_user = Uri.parse(“content://scut.carson_ho.myprovider/user”);
// 插入表中数据
ContentValues values = new ContentValues();
values.put(“_id”, 4);
values.put(“name”, “Jordan”);
// 获取ContentResolver
ContentResolver resolver = getContentResolver();
// 通过ContentResolver 根据URI 向ContentProvider中插入数据
resolver.insert(uri_user,values);
// 通过ContentResolver 向ContentProvider中查询数据
Cursor cursor = resolver.query(uri_user, new String[]{“_id”,“name”}, null, null, null);
while (cursor.moveToNext()){
System.out.println(“query book:” + cursor.getInt(0) +" "+ cursor.getString(1));
// 将表中数据全部输出
}
cursor.close();
// 关闭游标
/**
- 对job表进行操作
*/
// 和上述类似,只是URI需要更改,从而匹配不同的URI CODE,从而找到不同的数据资源
Uri uri_job = Uri.parse(“content://scut.carson_ho.myprovider/job”);
// 插入表中数据
ContentValues values2 = new ContentValues();
values2.put(“_id”, 4);
values2.put(“job”, “NBA Player”);
// 获取ContentResolver
ContentResolver resolver2 = getContentResolver();
// 通过ContentResolver 根据URI 向ContentProvider中插入数据
resolver2.insert(uri_job,values2);
// 通过ContentResolver 向ContentProvider中查询数据
Cursor cursor2 = resolver2.query(uri_job, new String[]{“_id”,“job”}, null, null, null);
while (cursor2.moveToNext()){
System.out.println(“query job:” + cursor2.getInt(0) +" "+ cursor2.getString(1));
// 将表中数据全部输出
}
cursor2.close();
// 关闭游标
}
}
结果展示
在进程展示时,需要先运行准备数据的进程1,再运行需要访问数据的进程2
- 运行准备数据的进程1
在进程1中,我们准备好了一系列数据
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
最后
目前已经更新的部分资料:
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
351)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
最后
目前已经更新的部分资料:
[外链图片转存中…(img-toamhKsD-1712333457351)]
[外链图片转存中…(img-XlSMxZe3-1712333457352)]
[外链图片转存中…(img-hlYWXnx7-1712333457352)]