`

编写最简单的Content Provider

阅读更多

来自:http://www.apkbus.com/android-16352-1-1.html

 

在自己的android应用中存储数据,可以用SQLite数据库。不过,如果需要在多个应用中共享数据,在Android中,只有通过Content provider机制。

下面用一个最简单(不完整)的示例来说明Content Provider的创建。在界面中使用这个示例显示的效果:

1.png

 

显示帝王姓名、登基年份和朝代。

 

在这个简单示例中,包括两部分:

  • 创建一个Content Provider;
  • 使用这个Content Provider(目前是在同一个应用中,可以在其他应用中以相同方式调用)。

创建一个简陋的Content Provider

content provider通过抽象一致的接口,供其他开发者使用Content provider,而实现Content provider,既可以用sqlite3这样的android内置数据库,也可以使用文件系统,甚至可以自己写其他的实现,只要实现规定的接口即可。

实现一个Content provider,需要做:

  • 继承ContentProvider,实现该类的几个抽象方法;
  • 在manifest文件中声明这个Content provider。

实现的Content provider代码:

 

  1. package com.easymorse.cp;
  2.  
  3. import android.content.ContentProvider; 
  4. import android.content.ContentValues; 
  5. import android.content.Context; 
  6. import android.database.Cursor; 
  7. import android.database.sqlite.SQLiteDatabase; 
  8. import android.database.sqlite.SQLiteStatement; 
  9. import android.net.Uri;
  10.  
  11. public class MyContentProvider extends ContentProvider {
  12.  
  13.     public static final Uri CONTENT_URI = Uri.parse("content://com.easymorse.cp.mycp");
  14.  
  15.     public static final String _ID = "id";
  16.  
  17.     public static final String NAME = "name";
  18.  
  19.     public static final String DYNASTY = "dynasty";
  20.  
  21.     public static final String START_YEAR = "start_year";
  22.  
  23.     private static SQLiteDatabase database;
  24.  
  25.     private static void createTablesIfNotExists() { 
  26.         database.execSQL("create table if not exists emperors(" 
  27.                 + " id integer primary key autoincrement," + " name text," 
  28.                 + "dynasty text," + "start_year text" + ");");
  29.  
  30.         SQLiteStatement statement = database 
  31.                 .compileStatement("insert into emperors(name,dynasty,start_year) values(?,?,?)"); 
  32.         int index = 1; 
  33.         statement.bindString(index++, "朱元璋"); 
  34.         statement.bindString(index++, "明"); 
  35.         statement.bindString(index++, "1398"); 
  36.         statement.execute();
  37.  
  38.         index = 1; 
  39.         statement.bindString(index++, "玄烨"); 
  40.         statement.bindString(index++, "清"); 
  41.         statement.bindString(index++, "1722"); 
  42.         statement.execute();
  43.  
  44.         statement.close(); 
  45.     }
  46.  
  47.     @Override 
  48.     public int delete(Uri uri, String selection, String[] selectionArgs) { 
  49.         return 0; 
  50.     }
  51.  
  52.     @Override 
  53.     public String getType(Uri uri) { 
  54.         return null; 
  55.     }
  56.  
  57.     @Override 
  58.     public Uri insert(Uri uri, ContentValues contentValues) { 
  59.         // TODO Auto-generated method stub 
  60.         return null; 
  61.     }
  62.  
  63.     @Override 
  64.     public boolean onCreate() { 
  65.         if (database == null) { 
  66.             database = this.getContext().openOrCreateDatabase("emperors", 
  67.                     Context.MODE_PRIVATE, null); 
  68.             createTablesIfNotExists(); 
  69.         } 
  70.         return database != null; 
  71.     }
  72.  
  73.     @Override 
  74.     public Cursor query(Uri uri, String[] projection, String selection, 
  75.             String[] selectionArgs, String sortOrder) { 
  76.         Cursor cursor = database.rawQuery("select * from emperors", null); 
  77.         return cursor; 
  78.     }
  79.  
  80.     @Override 
  81.     public int update(Uri uri, ContentValues contentValues, String selection, 
  82.             String[] selectionArgs) { 
  83.         // TODO Auto-generated method stub 
  84.         return 0; 
  85.     }
  86.  
  87. }
复制代码

需要实现的方法是:

 

 

  1. query() << 
  2. insert() 
  3. update() 
  4. delete() 
  5. getType() 
  6. onCreate()<<
复制代码

在本例中值实现了标注<<的方法。onCreate方法是创建Content provider时,android调用的。创建Content provider时,本例检查是否创建emperors表,没有就创建一个,并且插入了2条记录。query方法实现的很简单,只是返回表的所有结果集。做完这些还不够,还需要声明一些常量:

  1. public static final Uri CONTENT_URI = Uri.parse("content://com.easymorse.cp.mycp");
  2. public static final String _ID = "id";
  3. public static final String NAME = "name";
  4. public static final String DYNASTY = "dynasty";
  5. public static final String START_YEAR = "start_year";
复制代码

CONTENT_URI 是便于Content provider使用者引用的,下面使用Content provider的时候能看到。另外,要有一个_ID常量,值是具体表的id列名称。其他几个常量是表中列的对应名称常量。目前没有真的使用。在manifest文件中的声明:

  1. <application android:icon="@drawable/icon" android:label="@string/app_name"> 
  2.     <activity android:name=".UseContactActivity" android:label="@string/app_name"> 
  3.         <intent-filter> 
  4.             <action android:name="android.intent.action.MAIN" /> 
  5.             <category android:name="android.intent.category.LAUNCHER" /> 
  6.         </intent-filter> 
  7.     </activity> 
  8.    <provider android:name="com.easymorse.cp.MyContentProvider" 
  9.         android:authorities="com.easymorse.cp.mycp"></provider> 
  10. </application>
复制代码

这里的android:name是ContentProvider实现类的类名,android:authorities是CONTENT_URI的值。使用创建的Content Provider代码在一个Activity中,通过TextView显示出emperors表中的数据:

  1. package com.easymorse.cp;
  2. import android.app.Activity; 
  3. import android.database.Cursor; 
  4. import android.os.Bundle; 
  5. import android.widget.TextView;
  6. public class UseContactActivity extends Activity { 
  7.     /** Called when the activity is first created. */ 
  8.     @Override 
  9.     public void onCreate(Bundle savedInstanceState) { 
  10.         super.onCreate(savedInstanceState);
  11.         TextView textView = new TextView(this); 
  12.         textView.setText(getContentProviderValues()); 
  13.         this.setContentView(textView); 
  14.     }
  15.     private String getContentProviderValues() { 
  16.         StringBuilder builder = new StringBuilder();
  17.         Cursor cursor = managedQuery(MyContentProvider.CONTENT_URI, null, null, 
  18.                 null, null); 
  19.         while (cursor.moveToNext()) { 
  20.             builder 
  21.                     .append( 
  22.                             cursor.getString(cursor 
  23.                                     .getColumnIndex(MyContentProvider.NAME))) 
  24.                     .append(" | ") 
  25.                     .append( 
  26.                             cursor 
  27.                                     .getString(cursor 
  28.                                             .getColumnIndex(MyContentProvider.START_YEAR))) 
  29.                     .append(" | ") 
  30.                     .append( 
  31.                             cursor.getString(cursor 
  32.                                     .getColumnIndex(MyContentProvider.DYNASTY))) 
  33.                     .append("\n"); 
  34.         }
  35.         return builder.toString(); 
  36.     } 
  37. }
复制代码

主要方法是getContentProviderValues()。通过managedQuery()方法获取到Cursor对象。在方法调用的参数,只用到了MyContentProvider.CONTENT_URI,通过这个参数,android可定位到具体的ContentProvider并启动它(如果没有启动的话)。后面几个参数类似SQL的SELECT子句、WHERE子句和ORDER BY子句。都不写,将返回全部的结果(其实我这个例子中的Content provider也只实现了返回全部结果的逻辑)。源代码见:
<ignore_js_op style="word-wrap: break-word; "> Content Provider安卓巴士(源代码).rar (48.5 KB, 下载次数: 99)

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics