橘子味的心
标题:7.8 使用自定义ContentProvider

上一节教程我们学习了如何自定义 ContentProvider,接下来将讲解如何通过 ContentProvider 访问其他应用程序中的数据,并对数据进行更改。

实例 UseDbProvider 演示了通过我们之前建立的自定义 ContentProvider MyDbProvider 访问实例 MyDbDemo 中建立的 SQLite 数据库 mydb,并对其中的数据进行 CRUD 操作的过程。

实例 UseDbProvider 对 MyDbProvider 相关信息的访问是从 MyFriendsDB 类中获取的。

实例 UseDbProvider 的运行效果如图 2 所示。该视图和实例 MyDbDemo 一样,由 main.xml 和 list.xml 组成。

UseDbProvider界面
图 2  UseDbProvider界面

该实例实现的步骤如下:

1)在 Eclipse 中建立工程 UseDbProvider,定义包为“introduction.android.useDbprovider”,定义 Activity 为 UseDbCPActivity。

2)从工程 MyDbDemo 中导出 MyDbProvider 的信息描述类 MyFriendsDB。具体操作方法如下:
  • 右击工程 MyDbDemo,在弹出的菜单中选择 Export 选项,如图 3 所示。
  • 在弹出的对话框中选择导出类型为 Java|JAR file,单击 Next 按钮,如图 4 所示。
选择Export选项
图 3  选择Export选项
 
选择导出类型
图 4  选择导出类型
在弹出的对话框中选择导出资源为 MyFriendsDB.Java,导出的目标文件为“C:\MyDbProvider.jar”,单击 Finish 按钮。这样,就把 MyFriendsDB 类导出到 myProvider.jar 文件中,也就可以导出到其他文件中使用了,如图 5 所示。
导出过程
图 5  导出过程
3)在工程 MyFriendsDB 中导入 MyFriendsDB。右击 UseDbProvider,选择 Build Path | Add External Archives 选项(如图 6 所示),在弹出的对话框中选中 MyDbProvder.jar,即可将 MyFriendsDB 类导入工程中。
选择Add External Archives选项
图 6  选择Add External Archives选项

编写 UseDbCPActivity 类,通过 ContentResolver 完成对 MyDbProvider 的访问,进而完成对数据的操作。

实例 UseDbProvider 中 UseDbCPActivity.java 的代码如下:
  1. import android.app.Activity;
  2. import android.content.ContentUris;
  3. import android.content.ContentValues;
  4. import android.database.Cursor;
  5. import android.net.Uri;
  6. import android.os.Bundle;
  7. import android.util.Log;
  8. import android.view.View;
  9. import android.widget.AdapterView;
  10. import android.widget.Button;
  11. import android.widget.EditText;
  12. import android.widget.ListView;
  13. import android.widget.SimpleAdapter;
  14.  
  15. import android.widget.AdapterView.OnItemClickListener;
  16.  
  17. import java.util.ArrayList;
  18. import java.util.HashMap;
  19. import java.util.List;
  20. import java.util.Map;
  21.  
  22. public class UseDbCPActivity extends Activity {
  23. private List<Map<String, String>> data;
  24. private SimpleAdapter listAdapter;
  25. private ListView listview;
  26. private HashMap<String, String> item;
  27. private Button selBtn, addBtn, updBtn, delBtn;
  28. private EditText et_name;
  29. private EditText et_age;
  30. private EditText et_id;
  31.  
  32. /**
  33. * Called when the activity is first created.
  34. */
  35.  
  36. @Override
  37. public void onCreate(Bundle savedInstanceState) {
  38. super.onCreate(savedInstanceState);
  39. setContentView(R.layout.activity_main);
  40. et_name = (EditText) findViewById(R.id.et_name);
  41. et_age = (EditText) findViewById(R.id.et_age);
  42. et_id = (EditText) findViewById(R.id.et_id);
  43. listview = (ListView) findViewById(R.id.listView);
  44. selBtn = (Button) findViewById(R.id.bt_query);
  45. addBtn = (Button) findViewById(R.id.bt_add);
  46. updBtn = (Button) findViewById(R.id.bt_modify);
  47. delBtn = (Button) findViewById(R.id.bt_del);
  48. selBtn.setOnClickListener(new Button.OnClickListener() {
  49. @Override
  50. public void onClick(View v) {
  51. // TODO Auto-generated method stub
  52. if (et_id.getText().toString().equals(""))
  53. dbFindAll(MyFriendsDB.CONTENT_TYPE);
  54. else
  55. dbFindAll(MyFriendsDB.CONTENT_ITEM_TYPE);
  56. }
  57. });
  58.  
  59. addBtn.setOnClickListener(new Button.OnClickListener() {
  60. @Override
  61. public void onClick(View v) {
  62. // TODO Auto-generated method stub
  63. dbAdd(null);
  64. dbFindAll(MyFriendsDB.CONTENT_TYPE);
  65. }
  66. });
  67.  
  68. updBtn.setOnClickListener(new Button.OnClickListener() {
  69. @Override
  70. public void onClick(View v) {
  71. // TODO Auto-generated method stub
  72. dbUpdate(null);
  73. dbFindAll(MyFriendsDB.CONTENT_TYPE);
  74. }
  75. });
  76. delBtn.setOnClickListener(new Button.OnClickListener() {
  77. @Override
  78. public void onClick(View v) {
  79. // TODO Auto-generated method stub
  80. dbDel(-1);
  81. dbFindAll(MyFriendsDB.CONTENT_TYPE);
  82. }
  83. });
  84. data = new ArrayList<Map<String, String>>();
  85. dbFindAll(MyFriendsDB.CONTENT_TYPE);
  86. listview.setOnItemClickListener(new OnItemClickListener() {
  87. private String selId;
  88.  
  89. @Override
  90. public void onItemClick(AdapterView<?> parent, View v, int position, long id) {
  91. // TODO Auto-generated method stub
  92. Map<String, Object> listltem = (Map<String, Object>) listview.getItemAtPosition(position);
  93. et_name.setText((String) listltem.get("name"));
  94. et_age.setText((String) listltem.get("age"));
  95. et_id.setText((String) listltem.get("_id"));
  96. Log.i("UseDB", "id=" + selId);
  97. }
  98. });
  99. }
  100.  
  101. private void showList() {
  102. // TODO Auto-generated method stub
  103. listAdapter = new SimpleAdapter(this, data,
  104. R.layout.listview,
  105. new String[]{"_id", "name", "age"},
  106. new int[]{R.id.tvID, R.id.tvName, R.id.tvAge});
  107. listview.setAdapter(listAdapter);
  108. }
  109.  
  110. protected void dbDel(long iid) {
  111. // TODO Auto-generated method stub
  112. if (iid < 0) {
  113. String id = et_id.getText().toString().trim();
  114. if (id.equals("")) {
  115. Log.e("UseDB", "未指定更新数据");
  116. return;
  117. }
  118. iid = Long.parseLong(id);
  119. }
  120.  
  121. Uri uri = ContentUris.withAppendedId(MyFriendsDB.CONTENT_URI, iid);
  122. int i = this.getContentResolver().delete(uri, null, null);
  123. if (i > 0) {
  124. Log.i("UseDB", " 已删除数据id=" + iid);
  125. } else {
  126. Log.i("UseDB", "数据未删除。");
  127. }
  128.  
  129. }
  130.  
  131. protected void dbUpdate(ContentValues values) {
  132. // TODO Auto-generated method stub
  133. String id = et_id.getText().toString().trim();
  134. if (id.equals("")) {
  135. Log.e("UseDB", "未指定更新数据。");
  136. return;
  137. }
  138. Long selid = Long.parseLong(id);
  139. Uri uri = ContentUris.withAppendedId(MyFriendsDB.CONTENT_URI, selid);
  140. if (values == null) {
  141. values = new ContentValues();
  142. values.put("name", et_name.getText().toString().trim());
  143. values.put("age", et_age.getText().toString().trim());
  144. }
  145. int i = this.getContentResolver().update(uri, values, null, null);
  146. if (i > 0) {
  147. Log.i("UseDB", "已更新数据id=" + selid);
  148. } else {
  149. Log.e("UseDB", "数据更新失败!");
  150. }
  151. }
  152.  
  153. protected void dbAdd(ContentValues values) {
  154. // TODO Auto-generated method stub
  155. if (values == null) {
  156. values = new ContentValues();
  157. values.put("name", et_name.getText().toString().trim());
  158. values.put("age", et_age.getText().toString().trim());
  159. }
  160. Uri uri = this.getContentResolver().insert(MyFriendsDB.CONTENT_URI, values);
  161. if (uri == null) {
  162. Log.e("UseDB", "数据插入失败!");
  163. }
  164. }
  165.  
  166. protected void dbFindAll(String type) {
  167. // TODO Auto-generated method stub
  168. data.clear();
  169. Cursor cursor;
  170. Uri uri;
  171. if (type == MyFriendsDB.CONTENT_TYPE) {
  172. uri = MyFriendsDB.CONTENT_URI;
  173. } else {
  174. Long selid = Long.parseLong(et_id.getText().toString().trim());
  175. uri = ContentUris.withAppendedId(MyFriendsDB.CONTENT_URI, selid);
  176. Log.d("UseDB", uri.toString());
  177. }
  178. cursor = this.getContentResolver().query(uri, null, null, null, null);
  179. cursor.moveToFirst();
  180. while (!cursor.isAfterLast()) {
  181. String id = cursor.getString(0);
  182. String name = cursor.getString(1);
  183. String age = cursor.getString(2);
  184. item = new HashMap<String, String>();
  185. item.put("_id", id);
  186. item.put("name", name);
  187. item.put("age", age);
  188. data.add(item);
  189. cursor.moveToNext();
  190. }
  191. showList();
  192. }
  193. }
由于工程 MyDbDemo 中定义了 MyDbProvider 的访问权限,因此实例 UseDbProvider 的 AndroidManifest.xml 文件中也必须声明相应权限。

AndroidManifest.xml 文件的代码如下:
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="introduction.android.useDbdemo"
  4. android:versionCode="1"
  5. android:versionName="1.0">
  6.  
  7. <uses-sdk android:minSdkVersion="14" />
  8. <uses-permission android:name="introduction.android.permission.USE_MYDB" />
  9. <application
  10. android:allowBackup="true"
  11. android:icon="@mipmap/ic_launcher"
  12. android:label="@string/app_name"
  13. android:roundIcon="@mipmap/ic_launcher_round"
  14. android:supportsRtl="true"
  15. android:theme="@style/AppTheme">
  16. <activity android:name=".UseDbCPActivity">
  17. <intent-filter>
  18. <action android:name="android.intent.action.MAIN" />
  19. <category android:name="android.intent.category.LAUNCHER" />
  20. </intent-filter>
  21. </activity>
  22. </application>
  23.  
  24. </manifest>
在实例 UseDbProvider 中对 SQLite 数据库 mydb 进行 CRUD 操作后,运行 MyDbDemo 进行查询,可发现数据库中的数据确实被改变了。由此实现了在一个应用程序中通过自定义的 ContentProvider 修改另一个应用程序中的持久化数据的功能。