欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

android基礎總結(jié)篇之八:創(chuàng)建及調(diào)用自己的ContentProvider

 更新時間:2016年11月17日 14:45:12   作者:liuhe688  
這篇文章主要介紹了android基礎總結(jié)篇之八:創(chuàng)建及調(diào)用自己的ContentProvider,有興趣的可以了解一下。

今天我們來講解一下如何創(chuàng)建及調(diào)用自己的ContentProvider。

在前面兩篇文章中我們分別講了如何讀寫聯(lián)系人和短消息,相信大家對于ContentProvider的操作方法已經(jīng)有了一定程度的了解。在有些場合,除了操作ContentProvider之外,我們還有可能需要創(chuàng)建自己的ContentProvider,來提供信息共享的服務,這就要求我們很好的掌握ContentProvider的創(chuàng)建及使用技巧。下面我們就由表及里的逐步講解每個步驟。

在正式開始實例演示之前,我們先來了解以下兩個知識點:

授權:

在Android中,每一個ContentProvider都會用類似于域名的字符串來注冊自己,我們成為授權(authority)。這個唯一標識的字符串是此ContentProvider可提供的一組URI的基礎,有了這個基礎,才能夠向外界提供信息的共享服務。

授權是在AndroidManifest.xml中完成的,每一個ContentProvider必須在此聲明并授權,方式如下:

<provider android:name=".SomeProvider" 
  android:authorities="com.your-company.SomeProvider"/> 

上面的<provider>元素指明了ContentProvider的提供者是“SomeProvider”這個類,并為其授權,授權的基礎URI為“com.your-company.SomeProvider”。有了這個授權信息,系統(tǒng)可以準確的定位到具體的ContentProvider,從而使訪問者能夠獲取到指定的信息。這和瀏覽Web頁面的方式很相似,“SomeProvider”就像一臺具體的服務器,而“com.your-company.SomeProvider”就像注冊的域名,相信大家對這個概念并不陌生,由此聯(lián)想一下就可以了解ContentProvider授權的作用了。(需要注意的是,除了Android內(nèi)置應用程序之外,第三方程序應盡量使用以上方式的完全限定的授權名。)

MIME類型:

就像網(wǎng)站返回給定URL的MIME(Multipurpose Internet Mail Extensions,多用途Internet郵件擴展)類型一樣(這使瀏覽器能夠用正確的程序來查看內(nèi)容),ContentProvider還負責返回給定URI的MIME類型。根據(jù)MIME類型規(guī)范,MIME類型包含兩部分:類型和子類型。例如:text/html,text/css,text/xml等等。

Android也遵循類似的約定來定義MIME類型。

對于單條記錄,MIME類型類似于:

vnd.android.cursor.item/vnd.your-company.content-type

而對于記錄的集合,MIME類型類似于:

vnd.android.cursor.dir/vnd.your-company.comtent-type

其中的vnd表示這些類型和子類型具有非標準的、供應商特定的形式;content-type可以根據(jù)ContentProvider的功能來定,比如日記的ContentProvider可以為note,日程安排的ContentProvider可以為schedule,等等。

了解了以上兩個知識點之后,我們就結(jié)合實例來演示一下具體的過程。

我們將會創(chuàng)建一個記錄person信息的ContentProvider,實現(xiàn)對person的CRUD操作,訪問者可以通過下面路徑操作我們的ContentProvider:

訪問者可以通過“[BASE_URI]/persons”來操作person集合,也可以通過“[BASE_URI]/persons/#”的形式操作單個person。
我們創(chuàng)建一個person的ContentProvider需要兩個步驟:

1.創(chuàng)建PersonProvider類:

我們需要繼承ContentProvider類,實現(xiàn)onCreate、query、insert、update、delete和getType這幾個方法。具體代碼如下:

package com.scott.provider; 
 
import android.content.ContentProvider; 
import android.content.ContentUris; 
import android.content.ContentValues; 
import android.content.UriMatcher; 
import android.database.Cursor; 
import android.database.sqlite.SQLiteDatabase; 
import android.net.Uri; 
 
public class PersonProvider extends ContentProvider { 
 
  private static final UriMatcher matcher; 
  private DBHelper helper; 
  private SQLiteDatabase db; 
   
  private static final String AUTHORITY = "com.scott.provider.PersonProvider"; 
  private static final int PERSON_ALL = 0; 
  private static final int PERSON_ONE = 1; 
   
  public static final String CONTENT_TYPE = "vnd.android.cursor.dir/vnd.scott.person"; 
  public static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/vnd.scott.person"; 
   
  //數(shù)據(jù)改變后立即重新查詢 
  private static final Uri NOTIFY_URI = Uri.parse("content://" + AUTHORITY + "/persons"); 
   
  static { 
    matcher = new UriMatcher(UriMatcher.NO_MATCH); 
     
    matcher.addURI(AUTHORITY, "persons", PERSON_ALL);  //匹配記錄集合 
    matcher.addURI(AUTHORITY, "persons/#", PERSON_ONE); //匹配單條記錄 
  } 
   
  @Override 
  public boolean onCreate() { 
    helper = new DBHelper(getContext()); 
    return true; 
  } 
 
  @Override 
  public String getType(Uri uri) { 
    int match = matcher.match(uri); 
    switch (match) { 
    case PERSON_ALL: 
      return CONTENT_TYPE; 
    case PERSON_ONE: 
      return CONTENT_ITEM_TYPE; 
    default: 
      throw new IllegalArgumentException("Unknown URI: " + uri); 
    } 
  } 
   
  @Override 
  public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) { 
    db = helper.getReadableDatabase(); 
    int match = matcher.match(uri); 
    switch (match) { 
    case PERSON_ALL: 
      //doesn't need any code in my provider. 
      break; 
    case PERSON_ONE: 
      long _id = ContentUris.parseId(uri); 
      selection = "_id = ?"; 
      selectionArgs = new String[]{String.valueOf(_id)}; 
      break; 
    default: 
      throw new IllegalArgumentException("Unknown URI: " + uri); 
    } 
    return db.query("person", projection, selection, selectionArgs, null, null, sortOrder); 
  } 
 
  @Override 
  public Uri insert(Uri uri, ContentValues values) { 
    int match = matcher.match(uri); 
    if (match != PERSON_ALL) { 
      throw new IllegalArgumentException("Wrong URI: " + uri); 
    } 
    db = helper.getWritableDatabase(); 
    if (values == null) { 
      values = new ContentValues(); 
      values.put("name", "no name"); 
      values.put("age", "1"); 
      values.put("info", "no info."); 
    } 
    long rowId = db.insert("person", null, values); 
    if (rowId > 0) { 
      notifyDataChanged(); 
      return ContentUris.withAppendedId(uri, rowId); 
    } 
    return null; 
  } 
 
  @Override 
  public int delete(Uri uri, String selection, String[] selectionArgs) { 
    db = helper.getWritableDatabase(); 
    int match = matcher.match(uri); 
    switch (match) { 
    case PERSON_ALL: 
      //doesn't need any code in my provider. 
      break; 
    case PERSON_ONE: 
      long _id = ContentUris.parseId(uri); 
      selection = "_id = ?"; 
      selectionArgs = new String[]{String.valueOf(_id)}; 
    } 
    int count = db.delete("person", selection, selectionArgs); 
    if (count > 0) { 
      notifyDataChanged(); 
    } 
    return count; 
  } 
 
  @Override 
  public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) { 
    db = helper.getWritableDatabase(); 
    int match = matcher.match(uri); 
    switch (match) { 
    case PERSON_ALL: 
      //doesn't need any code in my provider. 
      break; 
    case PERSON_ONE: 
      long _id = ContentUris.parseId(uri); 
      selection = "_id = ?"; 
      selectionArgs = new String[]{String.valueOf(_id)}; 
      break; 
    default: 
      throw new IllegalArgumentException("Unknown URI: " + uri); 
    } 
    int count = db.update("person", values, selection, selectionArgs); 
    if (count > 0) { 
      notifyDataChanged(); 
    } 
    return count; 
  } 
 
  //通知指定URI數(shù)據(jù)已改變 
  private void notifyDataChanged() { 
    getContext().getContentResolver().notifyChange(NOTIFY_URI, null);     
  } 
} 

在PersonProvider中,我們定義了授權地址為“com.scott.provider.PersonProvider”,相信大家在前面也有所了解了?;谶@個授權,我們使用了一個UriMatcher對其路徑進行匹配,“[BASE_URI]/persons"和“[BASE_URI]/persons/#”這兩種路徑我們在上面也介紹過,分別對應記錄集合和單個記錄的操作。在query、insert、update和delete方法中我們根據(jù)UriMatcher匹配結(jié)果來判斷該URI是操作記錄集合還是單條記錄,從而采取不同的處理方法。在getType方法中,我們會根據(jù)匹配的結(jié)果返回不同的MIME類型,這一步是不能缺少的,比如我們在query方法中有可能是查詢?nèi)考?,有可能是查詢單條記錄,那么我們返回的Cursor或是集合類型,或是單條記錄,這個跟getType返回的MIME類型是一致的,就好像瀏覽網(wǎng)頁一樣,指定的url返回的信息是什么類型,那么瀏覽器就應該接收到對應的MIME類型。另外,我們注意到,上面代碼中,在insert、update、delete方法中都調(diào)用了notifyDataChanged方法,這個方法中僅有的一步操作就是通知“[BASE_URI]/persons"的訪問者,數(shù)據(jù)發(fā)生改變了,應該重新加載了。

在我們的PersonProvider中,我們用到了Person、DBHelper類,代碼如下:

package com.scott.provider; 
 
public class Person { 
  public int _id; 
  public String name; 
  public int age; 
  public String info; 
   
  public Person() { 
  } 
   
  public Person(String name, int age, String info) { 
    this.name = name; 
    this.age = age; 
    this.info = info; 
  } 
} 
[java] view plain copy
package com.scott.provider; 
 
import android.content.Context; 
import android.database.sqlite.SQLiteDatabase; 
import android.database.sqlite.SQLiteOpenHelper; 
 
public class DBHelper extends SQLiteOpenHelper { 
 
  private static final String DATABASE_NAME = "provider.db"; 
  private static final int DATABASE_VERSION = 1; 
   
  public DBHelper(Context context) { 
    super(context, DATABASE_NAME, null, DATABASE_VERSION); 
  } 
 
  @Override 
  public void onCreate(SQLiteDatabase db) { 
    String sql = "CREATE TABLE IF NOT EXISTS person" + 
        "(_id INTEGER PRIMARY KEY AUTOINCREMENT, name VARCHAR, age INTEGER, info TEXT)"; 
    db.execSQL(sql); 
  } 
 
  @Override 
  public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { 
    db.execSQL("DROP TABLE IF EXISTS person"); 
    onCreate(db); 
  } 
} 

最后,要想讓這個ContentProvider生效,我們需要在AndroidManifest.xml中聲明并為其授權,如下所示:

<provider android:name=".PersonProvider" 
  android:authorities="com.scott.provider.PersonProvider" 
  android:multiprocess="true"/> 

其中,android:multiprocess代表是否允許多進程操作。另外我們也可以為其聲明相應的權限,對應的屬性是:android:permission。

2.調(diào)用PersonProvider類:

完成了person的ContentProvider后,下面我們來看一下如何訪問它。這一步我們在MainActivity中完成,看下面代碼:

package com.scott.provider; 
 
import java.util.ArrayList; 
 
import android.app.Activity; 
import android.content.ContentResolver; 
import android.content.ContentUris; 
import android.content.ContentValues; 
import android.database.Cursor; 
import android.database.CursorWrapper; 
import android.net.Uri; 
import android.os.Bundle; 
import android.os.Handler; 
import android.os.Message; 
import android.view.View; 
import android.widget.ListView; 
import android.widget.SimpleCursorAdapter; 
 
 
public class MainActivity extends Activity { 
   
  private ContentResolver resolver; 
  private ListView listView; 
   
  private static final String AUTHORITY = "com.scott.provider.PersonProvider"; 
  private static final Uri PERSON_ALL_URI = Uri.parse("content://" + AUTHORITY + "/persons"); 
   
  private Handler handler = new Handler() { 
    public void handleMessage(Message msg) { 
      //update records. 
      requery(); 
    }; 
  }; 
   
  @Override 
  public void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.main); 
 
    resolver = getContentResolver(); 
    listView = (ListView) findViewById(R.id.listView); 
     
    //為PERSON_ALL_URI注冊變化通知 
    getContentResolver().registerContentObserver(PERSON_ALL_URI, true, new PersonObserver(handler)); 
  } 
   
  /** 
   * 初始化 
   * @param view 
   */ 
  public void init(View view) { 
    ArrayList<Person> persons = new ArrayList<Person>(); 
     
    Person person1 = new Person("Ella", 22, "lively girl"); 
    Person person2 = new Person("Jenny", 22, "beautiful girl"); 
    Person person3 = new Person("Jessica", 23, "sexy girl"); 
    Person person4 = new Person("Kelly", 23, "hot baby"); 
    Person person5 = new Person("Jane", 25, "pretty woman"); 
     
    persons.add(person1); 
    persons.add(person2); 
    persons.add(person3); 
    persons.add(person4); 
    persons.add(person5); 
 
    for (Person person : persons) { 
      ContentValues values = new ContentValues(); 
      values.put("name", person.name); 
      values.put("age", person.age); 
      values.put("info", person.info); 
      resolver.insert(PERSON_ALL_URI, values); 
    } 
  } 
   
  /** 
   * 查詢所有記錄 
   * @param view 
   */ 
  public void query(View view) { 
//   Uri personOneUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1);查詢_id為1的記錄 
    Cursor c = resolver.query(PERSON_ALL_URI, null, null, null, null); 
     
    CursorWrapper cursorWrapper = new CursorWrapper(c) { 
       
      @Override 
      public String getString(int columnIndex) { 
        //將簡介前加上年齡 
        if (getColumnName(columnIndex).equals("info")) { 
          int age = getInt(getColumnIndex("age")); 
          return age + " years old, " + super.getString(columnIndex); 
        } 
        return super.getString(columnIndex); 
      } 
    }; 
     
    //Cursor須含有"_id"字段 
    SimpleCursorAdapter adapter = new SimpleCursorAdapter(this, android.R.layout.simple_list_item_2, 
        cursorWrapper, new String[]{"name", "info"}, new int[]{android.R.id.text1, android.R.id.text2}); 
    listView.setAdapter(adapter); 
     
    startManagingCursor(cursorWrapper); //管理Cursor 
  } 
   
  /** 
   * 插入一條記錄 
   * @param view 
   */ 
  public void insert(View view) { 
    Person person = new Person("Alina", 26, "attractive lady"); 
    ContentValues values = new ContentValues(); 
    values.put("name", person.name); 
    values.put("age", person.age); 
    values.put("info", person.info); 
    resolver.insert(PERSON_ALL_URI, values); 
  } 
   
  /** 
   * 更新一條記錄 
   * @param view 
   */ 
  public void update(View view) { 
    Person person = new Person(); 
    person.name = "Jane"; 
    person.age = 30; 
    //將指定name的記錄age字段更新為30 
    ContentValues values = new ContentValues(); 
    values.put("age", person.age); 
    resolver.update(PERSON_ALL_URI, values, "name = ?", new String[]{person.name}); 
     
    //將_id為1的age更新為30 
//   Uri updateUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1); 
//   resolver.update(updateUri, values, null, null); 
  } 
   
  /** 
   * 刪除一條記錄 
   * @param view 
   */ 
  public void delete(View view) { 
    //刪除_id為1的記錄 
    Uri delUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1); 
    resolver.delete(delUri, null, null); 
     
    //刪除所有記錄 
//   resolver.delete(PERSON_ALL_URI, null, null); 
  } 
   
  /** 
   * 重新查詢 
   */ 
  private void requery() { 
    //實際操作中可以查詢集合信息后Adapter.notifyDataSetChanged(); 
    query(null); 
  } 
} 

我們看到,在上面的代碼中,分別對應每一種情況進行測試,相對較為簡單。我們主要講一下registerContentObserver這一環(huán)節(jié)。

在前面的PersonProvider我們也提到,在數(shù)據(jù)更改后,會向指定的URI訪問者發(fā)出通知,以便于更新查詢記錄。大家注意,僅僅是ContentProvider出力還不夠,我們還需要在訪問者中注冊一個ContentObserver,才能夠接收到這個通知。下面我們創(chuàng)建一個

PersonObserver:
package com.scott.provider; 
 
import android.database.ContentObserver; 
import android.os.Handler; 
import android.os.Message; 
import android.util.Log; 
 
public class PersonObserver extends ContentObserver { 
 
  public static final String TAG = "PersonObserver"; 
  private Handler handler; 
   
  public PersonObserver(Handler handler) { 
    super(handler); 
    this.handler = handler; 
  } 
   
  @Override 
  public void onChange(boolean selfChange) { 
    super.onChange(selfChange); 
    Log.i(TAG, "data changed, try to requery."); 
    //向handler發(fā)送消息,更新查詢記錄 
    Message msg = new Message(); 
    handler.sendMessage(msg); 
  } 
} 

這樣一來,當ContentProvider發(fā)來通知之后,我們就能立即接收到,從而向handler發(fā)送一條消息,重新查詢記錄,使我們能夠看到最新的記錄信息。

最后,我們要在AndroidManifest.xml中為MainActivity添加MIME類型過濾器,告訴系統(tǒng)MainActivity可以處理的信息類型:

<!-- MIME類型 --> 
<intent-filter> 
  <data android:mimeType="vnd.android.cursor.dir/vnd.scott.person"/> 
</intent-filter> 
<intent-filter> 
  <data android:mimeType="vnd.android.cursor.item/vnd.scott.person"/> 
</intent-filter> 

這樣就完成了訪問者的代碼,我們來看一下效果:

鑒于操作類型太多,我在這里就不再展示了,大家可以自己試一試。

原文鏈接:http://blog.csdn.net/liuhe688/article/details/7050868

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • android實現(xiàn)手機與單片機藍牙模塊通信

    android實現(xiàn)手機與單片機藍牙模塊通信

    這篇文章主要為大家詳細介紹了android實現(xiàn)手機與單片機藍牙模塊通信的方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-05-05
  • Android 判斷是否有外網(wǎng)連接

    Android 判斷是否有外網(wǎng)連接

    本文給大家分享的是使用Android實現(xiàn)判斷是否有外網(wǎng)鏈接,有需要的小伙伴可以參考下。
    2016-02-02
  • Android顯示系統(tǒng)SurfaceFlinger分析

    Android顯示系統(tǒng)SurfaceFlinger分析

    本文詳細講解了Android顯示系統(tǒng)SurfaceFlinger,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-12-12
  • Android動態(tài)添加View的問題解決方法

    Android動態(tài)添加View的問題解決方法

    Android動態(tài)添加View的問題解決方法,需要的朋友可以參考一下
    2013-05-05
  • Android開發(fā)手冊SeekBar拖動條使用實例

    Android開發(fā)手冊SeekBar拖動條使用實例

    這篇文章主要為大家介紹了Android開發(fā)手冊SeekBar拖動條使用實例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-06-06
  • Android仿微信網(wǎng)絡加載彈出框

    Android仿微信網(wǎng)絡加載彈出框

    這篇文章主要為大家詳細介紹了Android仿微信網(wǎng)絡加載彈出框,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-06-06
  • android利用service完成計時功能

    android利用service完成計時功能

    這篇文章主要為大家詳細介紹了android利用service完成計時功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • 源碼剖析Android中Okio的使用

    源碼剖析Android中Okio的使用

    這篇文章主要將從源碼出發(fā),帶大家剖析一下Android中Okio的具體使用,文中的示例代碼講解詳細,具有一定的借鑒價值,感興趣的小伙伴可以了解一下
    2023-02-02
  • Kotlin條件控制語句匯總講解

    Kotlin條件控制語句匯總講解

    條件控制是每門編程語言中必不可少的,一般就是使用我們所熟知的 ifelse ,來作為我們代碼邏輯選擇條件控制。 在 Java 中一般使用 ifelse 和 switch-case 來作為條件控制,而在 Kotlin 中則是使用 if-else 和 when 來作為條件控制
    2022-09-09
  • Android開發(fā)之SQLite的使用方法

    Android開發(fā)之SQLite的使用方法

    本篇文章介紹了,Android開發(fā)之SQLite的使用方法。需要的朋友參考下
    2013-04-04

最新評論