Android使用ContentProvider實(shí)現(xiàn)跨進(jìn)程通訊示例詳解
1 前言
ContentProvider 即內(nèi)容提供器,是 Android 四大組件之一,為 App 存取數(shù)據(jù)提供統(tǒng)一的對(duì)外接口,讓不同的應(yīng)用之間可以共享數(shù)據(jù)。
如圖,Server 端通過(guò) ContentProvider 對(duì)外提供操作本地?cái)?shù)據(jù)(DataBase、File 等)的接口,Client 端通過(guò) ContentResolver 與 ContentProvider 通訊,從而實(shí)現(xiàn)跨進(jìn)程操作 Server 端數(shù)據(jù),Observer 端通過(guò) ContentObserver 監(jiān)聽(tīng) Server 端的數(shù)據(jù)是否發(fā)生變化,并觸發(fā)相關(guān)操作。
(1) ContentProvider 接口
ContentProvider 是一個(gè)抽象類(lèi),用戶需要實(shí)現(xiàn)如下抽象方法。
// 創(chuàng)建數(shù)據(jù)庫(kù)并獲得數(shù)據(jù)庫(kù)連接 public abstract boolean onCreate() // 查詢(xún)數(shù)據(jù) public abstract Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) // 插入數(shù)據(jù) public abstract Uri insert(Uri uri, ContentValues values) // 刪除數(shù)據(jù) public abstract int delete(Uri uri, String selection, String[] selectionArgs) // 更新數(shù)據(jù) public abstract int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) // 獲取數(shù)據(jù)類(lèi)型(MIME類(lèi)型,如:"text/html"、"image/png"、"message/rfc882"、"vnd.android-dir/mms-sms") public abstract String getType(Uri uri)
參數(shù)說(shuō)明:
- uri:數(shù)據(jù)表路徑
- projection:需要查詢(xún)的字段名稱(chēng)
- selection:查詢(xún)條件
- selectionArgs:查詢(xún)條件中的參數(shù)列表
- sortOrder:排序
(2)ContentResolver 接口
// 查詢(xún)數(shù)據(jù) public final Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) // 插入數(shù)據(jù) public final Uri insert(Uri url, ContentValues values) // 刪除數(shù)據(jù) public final int delete(Uri url, String selection, String[] selectionArgs) // 更新數(shù)據(jù) public final int update(Uri uri, ContentValues values, String selection, String[] selectionArgs)
可以看到,ContentProvider 和 ContentResolver 都有增刪改查操作,并且參數(shù)列表完全一致,以實(shí)現(xiàn)對(duì) Server 端數(shù)據(jù)的操作。
(3)ContentObserver 接口
ContentObserver 是一個(gè)抽象類(lèi),用戶需要重寫(xiě)其構(gòu)造方法和 onChange() 方法。
//用戶需要重寫(xiě)構(gòu)造方法,并注入 handler public ContentObserver(Handler handler) //監(jiān)聽(tīng)的內(nèi)容發(fā)生變化時(shí)調(diào)用 public void onChange(boolean selfChange) //注冊(cè) ContentObserver mContext.getContentResolver().registerContentObserver(uri, false, observer); //注銷(xiāo) ContentObserver mContext.getContentResolver().unregisterContentObserver(observer);
2 URI 簡(jiǎn)介
URI 即統(tǒng)一資源標(biāo)識(shí)符(Uniform Resource Identifier),能夠唯一標(biāo)識(shí)資源,如同資源的身份ID。數(shù)據(jù)庫(kù)的 URI 在 ContentProvider 中定義,ContentResolver 通過(guò)資源 URI 找到對(duì)應(yīng)的 ContentProvider,從而操作數(shù)據(jù)庫(kù)。URI 類(lèi)圖如下。
URI 由3部分組成:scheme、authority、path,其中 authority 又由 host、port 組成,URI 一般格式如下:
scheme://authority/path?query scheme://host:port/path?query
URI 類(lèi)的常用接口如下:
//根據(jù) uriString 生成 StringUri 對(duì)象(Uri的子類(lèi),重寫(xiě)了Uri的抽象方法) public static Uri parse(String uriString) //根據(jù) fiel 生成 HierarchicalUri 對(duì)象(Uri的子類(lèi),重寫(xiě)了Uri的抽象方法) public static Uri fromFile(File file) //根據(jù) scheme、ssp、fragment 生成 OpaqueUri 對(duì)象(Uri的子類(lèi),重寫(xiě)了Uri的抽象方法) public static Uri fromParts(String scheme, String ssp, String fragment) public abstract String getScheme() public abstract String getHost() public abstract int getPort() public abstract String getPath() public abstract String getQuery()
常見(jiàn)的 URI 實(shí)例如下:
//網(wǎng)絡(luò)資源 https://blog.csdn.net/m0_37602827 //本地資源 file:///sdcard/Pictures/WeiXin/a.mp4 //ContentProvider content://com.zhyan8.content.MyProvider/query //打電話 tel:10086 //發(fā)短信 smsto:10086 //發(fā)郵件 mailto:xxxx@qq.com //定位 geo:52.76,-79.0342
3 項(xiàng)目結(jié)構(gòu)
本文將以一個(gè)案例講解使用 ContentProvider 實(shí)現(xiàn)跨進(jìn)程通訊,項(xiàng)目結(jié)構(gòu)如下,一共包含 3 個(gè) Module。
- content_s:服務(wù)端,數(shù)據(jù)持有者,對(duì)外提供 ContentProvider
- content_c:客戶端,數(shù)據(jù)操作者,通過(guò) ContentResolver 操作服務(wù)端數(shù)據(jù)
- content_o:數(shù)據(jù)監(jiān)聽(tīng)者,通過(guò) ContentObserver 監(jiān)聽(tīng)服務(wù)端數(shù)據(jù)的變化
4 服務(wù)端(Content_S)
(1)數(shù)據(jù)庫(kù)操作類(lèi)
DBHelper.java
package com.zhyan8.content_s; import android.content.ContentValues; import android.content.Context; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteOpenHelper; public class DBHelper extends SQLiteOpenHelper { private static final String DATABASE = "test.db"; private static final String TABLE = "user"; public DBHelper(Context context) { super(context, DATABASE, null, 1); } @Override public void onCreate(SQLiteDatabase db) { String create_table = "create table " + TABLE + "(id int primary key, name varchar not null)"; db.execSQL(create_table); db.execSQL("insert into " + TABLE + " values(1001, '張三'),(1002, '李四')"); } @Override public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {} public Cursor query() { SQLiteDatabase db = getReadableDatabase(); String sql = "select * from " + TABLE; Cursor cursor = db.rawQuery(sql, null); return cursor; } public long insert(ContentValues cv) { SQLiteDatabase db = getWritableDatabase(); long result = db.insert(TABLE, null, cv); db.close(); return result; } }
生成的數(shù)據(jù)庫(kù)如下:
(2)自定義 ContentProvider
MyProvider.java
package com.zhyan8.content_s; import android.content.ContentProvider; import android.content.ContentValues; import android.content.UriMatcher; import android.database.Cursor; import android.net.Uri; import android.util.Log; public class MyProvider extends ContentProvider { private DBHelper mUserDBHelper; private static UriMatcher sUriMatcher = new UriMatcher(UriMatcher.NO_MATCH); //用于匹配URI,并返回對(duì)應(yīng)的操作編碼 private static final String AUTHORITES = "com.zhyan8.content.MyProvider"; private static final int QUERY = 1; //查詢(xún)操作編碼 private static final int INSERT = 2; //插入操作編碼 static { //添加有效的 URI 及其編碼 sUriMatcher.addURI(AUTHORITES, "/query", QUERY); sUriMatcher.addURI(AUTHORITES, "/insert", INSERT); } @Override public boolean onCreate() { mUserDBHelper = new DBHelper(getContext()); return false; } @Override public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) { Log.e("xxx-Provider", "query: "); int code = sUriMatcher.match(uri); if (code == QUERY) { return mUserDBHelper.query(); } return null; } @Override public Uri insert(Uri uri, ContentValues values) { Log.e("xxx-Provider", "insert: "); int code = sUriMatcher.match(uri); if (code == INSERT) { mUserDBHelper.insert(values); } getContext().getContentResolver().notifyChange(uri, null); //通知外界,數(shù)據(jù)發(fā)生變化 return null; } @Override public String getType(Uri uri) { //獲取資源 MIME return null; } @Override public int delete(Uri uri, String selection, String[] selectionArgs) { return 0; } @Override public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) { return 0; } }
(3)配置
在 AndroidManifest.xml 的 application 節(jié)點(diǎn)下配置 ContentProvider,如下。
<provider android:name=".MyProvider" android:authorities="com.zhyan8.content.MyProvider" android:exported="true" />
exported 屬性用于定義是否允許外界訪問(wèn)。
(4)主類(lèi)
MainActivity.java
package com.zhyan8.content_s; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); } }
5 客戶端(Content_C)
(1)數(shù)據(jù)類(lèi)
User.java
package com.zhyan8.content_c; public class User { private Integer id; private String name; public User(){}; public User(Integer id, String name) { this.id = id; this.name = name; } public void setId(Integer id) { this.id = id; } public void setName(String name) { this.name = name; } public Integer getId() { return id; } public String getName() { return name; } @Override public String toString() { return "User{" + "id=" + id + ", name='" + name + '\'' + '}'; } }
(2)自定義 ContentResolver
MyResolver.java
package com.zhyan8.content_c; import android.content.ContentResolver; import android.content.ContentValues; import android.content.Context; import android.database.ContentObserver; import android.database.Cursor; import android.net.Uri; import java.util.ArrayList; public class MyResolver { private ContentResolver mContentResolver; private static final String AUTHORITES = "com.zhyan8.content.MyProvider"; public MyResolver(Context context) { mContentResolver = context.getContentResolver(); } public ArrayList<User> query() { ArrayList<User> list = new ArrayList<>(); Uri uri = Uri.parse("content://" + AUTHORITES + "/query"); Cursor cursor = mContentResolver.query(uri, null, null, null, null); while (cursor.moveToNext()) { User user = new User(); user.setId(cursor.getInt(0)); user.setName(cursor.getString(1)); list.add(user); } cursor.close(); return list; } public void insert(User user) { Uri uri = Uri.parse("content://" + AUTHORITES + "/insert"); ContentValues cv = new ContentValues(); cv.put("id", user.getId()); cv.put("name", user.getName()); mContentResolver.insert(uri, cv); } }
(3)主類(lèi)
MainActivity.java
package com.zhyan8.content_c; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; import android.util.Log; import android.view.View; import java.util.ArrayList; public class MainActivity extends AppCompatActivity { private MyResolver myResolver; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); myResolver = new MyResolver(this); } public void clickQuery(View v) { ArrayList<User> users = myResolver.query(); Log.e("xxx", "clickQuery: " + users.toString()); } public void clickInsert(View v) { User user = new User(1003, "王五"); myResolver.insert(user); } }
(4)布局
activity_main.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity" android:orientation="vertical"> <Button android:layout_width="match_parent" android:layout_height="wrap_content" android:text="查詢(xún)" android:textSize="40sp" android:layout_marginTop="50dp" android:onClick="clickQuery"/> <Button android:layout_width="match_parent" android:layout_height="wrap_content" android:text="添加" android:textSize="40sp" android:layout_marginTop="50dp" android:onClick="clickInsert"/> </LinearLayout>
界面如下:
5 監(jiān)聽(tīng)者(Content_O)
(1)自定義 ContentObserver
MyObserver.java
package com.zhyan8.content_o; import android.database.ContentObserver; import android.os.Handler; public class MyObserver extends ContentObserver { private Uri mUri = Uri.parse("content://com.zhyan8.content.MyProvider/insert"); private ContentResolver mResolver; Handler mHandler; public MyObserver(Context context, Handler handler) { super(handler); mResolver = context.getContentResolver(); mHandler = handler; } @Override public void onChange(boolean selfChange) { mHandler.sendEmptyMessage(0x111); } public void registerObserver() { mResolver.registerContentObserver(mUri, false, this); } public void unregisterObserver() { mResolver.unregisterContentObserver(this); } }
(2)主類(lèi)
MainActivity.java
package com.zhyan8.content_o; import android.net.Uri; import android.os.Bundle; import android.os.Handler; import android.os.Message; import android.support.v7.app.AppCompatActivity; import android.util.Log; public class MainActivity extends AppCompatActivity { private MyObserver mObserver; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mObserver = new MyObserver(mHandler); mObserver.registerObserver(); } private Handler mHandler = new Handler() { @Override public void handleMessage(Message msg) { if (msg.what==0x111) { Log.e("xxx", "監(jiān)聽(tīng)的數(shù)據(jù)改變了"); } } }; @Override public void onDestroy() { mObserver.unregisterObserver(); super.onDestroy(); } }
6 效果展示
首先保證 Content_S、Content_C 和 Content_O 3個(gè)應(yīng)用都打開(kāi)了,再在 Content_C 端依次點(diǎn)擊【查詢(xún)】→【插入】→【查詢(xún)】,打印日志如下:
2020-11-22 22:47:41.397 5227-5597/com.zhyan8.content_s E/xxx-Provider: query:
2020-11-22 22:47:41.401 6261-6261/com.zhyan8.content_c E/xxx: clickQuery: [User{id=1001, name='張三'}, User{id=1002, name='李四'}]
2020-11-22 22:47:49.848 5227-5493/com.zhyan8.content_s E/xxx-Provider: insert:
2020-11-22 22:47:49.855 7879-7879/com.zhyan8.content_o E/xxx: 監(jiān)聽(tīng)的數(shù)據(jù)改變了
2020-11-22 22:47:52.055 5227-5597/com.zhyan8.content_s E/xxx-Provider: query:
2020-11-22 22:47:52.063 6261-6261/com.zhyan8.content_c E/xxx: clickQuery: [User{id=1001, name='張三'}, User{id=1002, name='李四'}, User{id=1003, name='王五'}]
以上就是Android使用ContentProvider實(shí)現(xiàn)跨進(jìn)程通訊示例詳解的詳細(xì)內(nèi)容,更多關(guān)于Android ContentProvider通訊的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Android notifyDataSetChanged() 動(dòng)態(tài)更新ListView案例詳解
這篇文章主要介紹了Android notifyDataSetChanged() 動(dòng)態(tài)更新ListView案例詳解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08Android中封裝RecyclerView實(shí)現(xiàn)添加頭部和底部示例代碼
這篇文章主要給大家介紹了關(guān)于Android中封裝RecyclerView實(shí)現(xiàn)添加頭部和底部的相關(guān)資料,網(wǎng)上這方面的資料很多,但都不是自己需要的,索性自己寫(xiě)一個(gè)分享出來(lái)供大家參考學(xué)習(xí),需要的朋友們下面隨著小編一起來(lái)學(xué)習(xí)學(xué)習(xí)吧。2017-08-08Android利用ContentProvider獲取本地?cái)?shù)據(jù)的方法
這篇文章主要介紹了Android利用ContentProvider獲取本地?cái)?shù)據(jù)的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-04-04android通過(guò)藍(lán)牙接收文件打開(kāi)時(shí)無(wú)法自動(dòng)選擇合適的應(yīng)用程序
android 通過(guò)藍(lán)牙接收文件,從歷史傳輸記錄打開(kāi),無(wú)法自動(dòng)選擇合適的應(yīng)用程序,比如video player打開(kāi).3gp、.mp4文件等等2013-06-06Android中webview與JS交互、互調(diào)方法實(shí)例詳解
這篇文章主要介紹了Android中webview與JS交互、互調(diào)方法實(shí)例詳解的相關(guān)資料,需要的朋友可以參考下2017-03-03Android的HTTP類(lèi)庫(kù)Volley入門(mén)學(xué)習(xí)教程
這篇文章主要介紹了Android應(yīng)用開(kāi)發(fā)框架Volley的入門(mén)學(xué)習(xí)教程,Volley適合于輕量級(jí)的通信功能開(kāi)發(fā),善于處理JSON對(duì)象,需要的朋友可以參考下2016-02-02Android之Notification的多種用法實(shí)例
本篇文章主要介紹了Android之Notification的多種用法實(shí)例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-12-12Android進(jìn)階之Spinner下拉框的使用方法
這篇文章主要為大家詳細(xì)介紹了Android進(jìn)階之Spinner下拉框的使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12Android 坐標(biāo)系與視圖坐標(biāo)系圖解分析
下面小編就為大家?guī)?lái)一篇Android 坐標(biāo)系與視圖坐標(biāo)系圖解分析。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-03-03android自定義控件實(shí)現(xiàn)簡(jiǎn)易時(shí)間軸(1)
這篇文章主要為大家詳細(xì)介紹了android自定義控件實(shí)現(xiàn)簡(jiǎn)易時(shí)間軸,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01