+4

Giới thiệu về Content Provider trong Android

Giới thiệu

Content provider là một thành phần để quản lý truy cập dữ liệu, nó cung cấp các phương thức khác nhau để các ứng dụng có thể truy cập dữ liệu từ một ứng dụng khác bằng cách sử dụng ContentResolver. Content Provider có thể giúp cho một ứng dụng quản lý quyền truy cập đến dữ liệu được lưu bởi ứng dụng đó, hoặc các ứng dụng khác, và đó là một cách để ta có thể chia sẻ dữ liệu cho các ứng dụng khác nhau. Hình dưới đây biểu diễn cho việc cách content providers quản lý việc truy cập tới bộ nhớ

Content Provider điều phối việc truy cập tới bỗ lưu trữ dữ liệu thông qua các API và các component như hình dưới, nó bao gồm

  • Chia sẻ dữ liệu từ ứng dụng của bán tới các ứng dụng khác
  • Gửi dữ liệu sang widget
  • Trả về một kết quả gợi ý khi search cho ứng dụng của bạn thông qua Seach Framework sử dụng SearchRecentSuggestionsProvider
  • Đồng bộ dữ liệu của ứng dụng với server bằng cách sử dụng AbstractThreadedSyncAdapter
  • Tải dữ liệu lên UI sử dụng CursorLoader

Content Provider hoạt động rất giống với một cơ sở dữ liệu, bạn có thể truy vấn, chỉnh sửa nội dung, cũng như là thêm xóa các nội dung sử dụng các phương thức: insert(), update(), delete(), query().

Sử dụng Content Provider

Để sử dụng Content Provider ta làm theo các bước sau:

  1. Xác định kiểu dữ liệu
  2. Xác định Uniform Resource Identifier (URI)
  3. Khai báo Content Provider trong manifest
  4. Implement lớp ContentProvider và các phương thức được yêu cầu.

Các phương thức cần được Override trong lớp Content Provider:

  • onCreate(): Phương thức này được gọi khi Provider được bắt đầu, nếu quá trình khởi tạo thành công trả về true, ngược lại là false
  • query(): Phương thức nhận yêu cầu từ Client. Kết quả được trả về như một đối tượng Cursor.
  • insert(): Phương thức chèn một dòng dữ liệu mới vào Content Provider.
  • delete(): Phương thức xóa một dòng dữ liệu đã tồn tại.
  • update(): Phương thức cập nhật một dòng dữ liệu nào đó đã tồn tại.
  • getType(): Phương thức trả về kiểu MIME của dữ liệu tại các URI.

Bạn có thể xem đoạn code sau mình sẽ sử dụng để mở CSDL ở bảng Student

public class SampleProvider extends ContentProvider {
    private SampleDBHelper mDBHelper;
    private SQLiteDatabase mDatabase;

    @Override
    public boolean onCreate() {
        // Khoi tao SampleDBHelper o ham onCreate, voi context lay tu method getContext()
        mDBHelper = new SampleDBHelper(getContext());
        return true;
    }

    @Nullable
    @Override
    public Cursor query(@NonNull Uri uri, @Nullable String[] projection, @Nullable String selection, @Nullable String[] selectionArgs, @Nullable String sortOrder) {
        mDatabase = mDBHelper.getReadableDatabase();
        // Truy van CSDL
        return mDatabase.query(SampleDBHelper.TABLE_STUDENT, projection, selection,
                selectionArgs, null, null, sortOrder);
    }

    @Nullable
    @Override
    public String getType(@NonNull Uri uri) {
        return null;
    }

    @Nullable
    @Override
    public Uri insert(@NonNull Uri uri, @Nullable ContentValues contentValues) {
        return null;
    }

    @Override
    public int delete(@NonNull Uri uri, @Nullable String s, @Nullable String[] strings) {
        return 0;
    }

    @Override
    public int update(@NonNull Uri uri, @Nullable ContentValues contentValues, @Nullable String s, @Nullable String[] strings) {
        return 0;
    }
}

Content URI

Content URI là một URI định danh dữ liệu trong một provider. Content URI bao gồm kí hiệu tên của toàn bộ provider và một tên chỉ tới một bảng. Khi bạn gọi một phương thức truy cập tới bảng trong provider thì Content URI của bảng đó là sẽ là một tham số. Để truy vấn data qua provider, ta sẽ sử dụng URI có định dạng như sau:

content://authority/path/id
  • content: luôn là content://
  • authority: một xâu để xác định tên của Content Provider, ví dụ như contact, browser,...
  • path: có thể không có hoặc được chia làm nhiều phần, và được phân cách bằng gạch chéo "/", dùng để định ra các thành phần con của dữ liệu. Ví dụ như để lấy danh sách các liên hệ trong danh bạ thì URI sẽ là content://contacts/people.
  • id: chỉ định rõ một bản ghi trong tập hợp dữ liệu, mỗi bản ghi sẽ được đánh dấu id là một số duy nhất.

Một số ví dụ về URI:

  • Content://media/internal/images – trả về List tất cả các ảnh lưu trong thiết bị
  • Content://contact/people – Trả về list tất cả các tên trong danh bạ
  • Content://contact/people/45 – Trả về một kết quả danh bạ có ID là 45

Trong bài này, mình sẽ hướng dẫn các bạn tự tạo một Content URI và sử dụng nó cùng với Content Provider để truy xuất dữ liệu từ CSDL nhé:

1. Tạo Database class

Đầu tiên, bạn cần tạo một SQLite Database, đầu tiên ta khởi tạo các giá trị cài đặt của DB như phiên bản database, database name, sau đó update ở trong constructor sử dụng các giá trị vừa khởi tạo đó như sau:

public class TutListDatabase extends SQLiteOpenHelper {
    private static final String DEBUG_TAG = "TutListDatabase";
    private static final int DB_VERSION = 1;
    private static final String DB_NAME = "tutorial_data";
 
    public TutListDatabase(Context context) {
        super(context, DB_NAME, null, DB_VERSION);
    }
 
    @Override
    public void onCreate(SQLiteDatabase db) {
    }
 
    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
    }

2. Định dạng Database Schema

Giả sử dữ liệu của chúng ta có các trường: title và url và id, đoạn code sau sẽ khai báo tạo một bảng với các thông số trên

public static final String TABLE_TUTORIALS = "tutorials";
public static final String ID = "_id";
public static final String COL_TITLE = "title";
public static final String COL_URL = "url";
 
private static final String CREATE_TABLE_TUTORIALS = "create table " + TABLE_TUTORIALS
+ " (" + ID + " integer primary key autoincrement, " + COL_TITLE
+ " text not null, " + COL_URL + " text not null);";
 
private static final String DB_SCHEMA = CREATE_TABLE_TUTORIALS;

3. Tạo Database

Để tạo DB, bên trong hàm onCreate(). đã sẽ chạy đoạn string DB_SCHEMA như là một câu lệnh SQL để tạo ra một bảng theo như những gì ta đã định dạng ở bước trước

@Override
public void onCreate(SQLiteDatabase db) {
    db.execSQL(DB_SCHEMA);
}

4. Tạo lớp Content Provider

Ta sẽ tạo một lớp TutListProvider kế thừa ContentProvider và tạo một instance của TutListDatabase ở bên trong hàm onCreate():

public class TutListProvider extends ContentProvider {
    private TutListDatabase mDB;
 
    @Override
    public boolean onCreate() {
        mDB = new TutListDatabase(getContext());
        return true;
}

5. Định nghĩa URI và các constant

Ví dụ, ta sẽ tạo ra một URI với tên và định dạng như sau

content:// com.mamlambo.tutorial.tutlist.data.TutListProvider/tutorials

Để sử dụng URI này, ta sẽ cần định nghĩa một vài constant để xác định dữ liệu nào ta sẽ truy xuất:

private static final String AUTHORITY = "com.mamlambo.tutorial.tutlist.data.TutListProvider";
public static final int TUTORIALS = 100;
public static final int TUTORIAL_ID = 110;
 
private static final String TUTORIALS_BASE_PATH = "tutorials";
public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY
        + "/" + TUTORIALS_BASE_PATH);
 
public static final String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE
        + "/mt-tutorial";
public static final String CONTENT_TYPE = ContentResolver.CURSOR_DIR_BASE_TYPE
        + "/mt-tutorial";

Để xác định kiểu của địa chỉ URI được truyền vào content provider, bạn có thể tạo class UriMatcher để xác định chính xác URI pattern nào được Content provider hỗ trợ, UriMatcher

private static final UriMatcher sURIMatcher = new UriMatcher(
        UriMatcher.NO_MATCH);
static {
    sURIMatcher.addURI(AUTHORITY, TUTORIALS_BASE_PATH, TUTORIALS);
    sURIMatcher.addURI(AUTHORITY, TUTORIALS_BASE_PATH + "/#", TUTORIAL_ID);
}

6. Xử lý các câu lệnh truy xuất

Content provider có một vài method cần được kế thừa. Nhưng trong trường hợp này ta chỉ cần sử dụng đến hàm query():

private static final UriMatcher sURIMatcher = new UriMatcher(
        UriMatcher.NO_MATCH);
static {
    sURIMatcher.addURI(AUTHORITY, TUTORIALS_BASE_PATH, TUTORIALS);
    sURIMatcher.addURI(AUTHORITY, TUTORIALS_BASE_PATH + "/#", TUTORIAL_ID);
}

7. Đăng kí Content Provider

Ta sẽ cần đăng kí provider trong manifest tương tự như activity

<provider
    android:authorities="com.mamlambo.tutorial.tutlist.data.TutListProvider"
    android:multiprocess="true"
    android:name="com.mamlambo.tutorial.tutlist.data.TutListProvider"></provider>

Tham khảo: https://developer.android.com/guide/topics/providers/content-provider-creating.html https://developer.android.com/reference/android/content/ContentUris.html https://code.tutsplus.com/tutorials/android-fundamentals-working-with-content-providers--mobile-5549


All rights reserved

Viblo
Hãy đăng ký một tài khoản Viblo để nhận được nhiều bài viết thú vị hơn.
Đăng kí