AndroidAZ系列有以下目的:
- Android程序猿的面试(初级,中级,高级,资深),拿到满意的offer。
- Android程序猿学习进阶。
标记说明:因为笔者是列出所有的Android知识点,因此面试不需要看那么多内容,如果是面试的知识点。笔者会加上标记Face,而如果不是面试的知识点,笔者会加上No标记,它是要学的东西;然后笔者将Android面试者或者面试者分为4个等级,初级A1,中级A2,高级A3,资深A4,如果这个知识点是所有等级的范围,那么笔者将会以all标记上。因此进阶路线就是A1->A2->A3->A4。也是面试者挑选的复习范围,假如你是中级程序员,那么你面试要看的内容就是包含A2&Face的标记。
All : 所有的Android工程师都看。
A1: 初级Android工程师。
A2: 中级Android工程师。
A3: 高级Android工程师。
A4: 资深Android工程师。
Face: 是面试的知识点。
No: 面试基本遇不到。
1.ContentProvider是什么
ContentProvider一般为存储和获取数据提供统一的接口,可以在不同的应用程序之间共享数据.
- 1,ContentProvider提供了对底层数据存储方式的抽象。比如下图中,底层使用了SQLite数据库,在用了ContentProvider封装后,即使你把数据库换成MongoDB,也不会对上层数据使用层代码产生影响。
Android框架中的一些类需要ContentProvider类型数据。如果你想让你的数据可以使用在如SyncAdapter, Loader, CursorAdapter等类上,那么你就需要为你的数据做一层ContentProvider封装。
第三个原因也是最主要的原因,是ContentProvider为应用间的数据交互提供了一个安全的环境。它准许你把自己的应用数据根据需求开放给其他应用进行增、删、改、查,而不用担心直接开放数据库权限而带来的安全问题。
2.ContentProvider的使用
Content Provider的用法有两种,一种是使用现有的content provider来读取和操作相应程序中的数据,另一种是创建自己的content provider提供外部访问接口.
要像访问content provider中共享的数据,要借助content resolver类,content resolver提供了一系列的方法对数据进行增删改查操作,以URI为参数.
2.1 URI
定义:Uniform Resource Identifier,即统一资源标识符
作用: 这里用来标识 ContentProvider 和其中的数据,外界进程通过 URI 找到对应的ContentProvider 和其中的数据,再进行数据操作
具体使用:URI分为 系统预置 & 自定义,分别对应系统内置的数据(如通讯录、日程表等等)和自定义数据库
``` java
// 设置URI
Uri uri = Uri.parse("content://com.carson.provider/User/1")
// 上述URI指向的资源是:名为 `com.carson.provider`的`ContentProvider` 中表名 为`User` 中的 `id`为1的数据
// 特别注意:URI模式存在匹配通配符* & #
// *:匹配任意长度的任何有效字符的字符串
// 以下的URI 表示 匹配provider的任何内容
content://com.example.app.provider/*
// #:匹配任意长度的数字字符的字符串
// 以下的URI 表示 匹配provider中的table表的所有行
content://com.example.app.provider/table/#
```
2.2 ContentResolver使用方法
``` java
// 使用ContentResolver前,需要先获取ContentResolver
// 可通过在所有继承Context的类中 通过调用getContentResolver()来获得ContentResolver
ContentResolver resolver = getContentResolver();
// 设置ContentProvider的URI
Uri uri = Uri.parse("content://cn.scu.myprovider/user");
// 根据URI 操作 ContentProvider中的数据
// 此处是获取ContentProvider中 user表的所有记录
Cursor cursor = resolver.query(uri, null, null, null, "userid desc");
```
3.如何创建自定义ContentProvider
首先我们创建一个自己的TestProvider继承ContentProvider。默认该Provider需要实现如下六个方法,
onCreate()
,
query(Uri, String[], String, String[], String)
,
insert(Uri, ContentValues)
,
update(Uri, ContentValues, String, String[])
,
delete(Uri, String, String[])
,
getType(Uri)
,
方法的具体介绍可以参考 官网
下面我们以实现insert和query方法为例
private final static int TEST = 100;
static UriMatcher buildUriMatcher() {
final UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH);
final String authority = TestContract.CONTENT_AUTHORITY;
matcher.addURI(authority, TestContract.PATH_TEST, TEST);
return matcher;
}
@Nullable
@Override
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
final SQLiteDatabase db = mOpenHelper.getReadableDatabase();
Cursor cursor = null;
switch ( buildUriMatcher().match(uri)) {
case TEST:
cursor = db.query(TestContract.TestEntry.TABLE_NAME, projection, selection, selectionArgs, sortOrder, null, null);
break;
}
return cursor;
}
@Nullable
@Override
public Uri insert(Uri uri, ContentValues values) {
final SQLiteDatabase db = mOpenHelper.getWritableDatabase();
Uri returnUri;
long _id;
switch ( buildUriMatcher().match(uri)) {
case TEST:
_id = db.insert(TestContract.TestEntry.TABLE_NAME, null, values);
if ( _id > 0 )
returnUri = TestContract.TestEntry.buildUri(_id);
else
throw new android.database.SQLException("Failed to insert row into " + uri);
break;
default:
throw new android.database.SQLException("Unknown uri: " + uri);
}
return returnUri;
}
此例中我们可以看到,我们根据path的不同,来区别对不同的数据库表进行操作,从而完成uri与具体数据库间的映射关系。
因为ContentProvider作为四大组件之一,所以还需要在AndroidManifest.xml中注册一下。
<provider
android:authorities="me.pengtao.contentprovidertest"
android:name=".provider.TestProvider"
/>
4.ContentProvider原理
使用binder在不同程序间传递数据.