`
fosa0989
  • 浏览: 110239 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

从Notepad实例学习Content Providers

阅读更多

Content providers是程序间共享数据的唯一方法。Android在 android.provider 包下提供了一些访问音频,视频,图像,个人联系信息等常用数据的Content providers,我们可以直接使用这些类来访问这些数据。如果想要分享自己的数据给别人,需要实现自定义的Content Provider,这通过继承类 ContentProvider 来实现。通过学习SDK下面Notepad的源码来学习如何自定义Content Provider。

 创建Content Provider分三个步骤:

1. 建立一个存储数据的系统,android中的大多数都是使用SQLite数据库(本文不涉及其它方式)

2. 扩展 ContentProvider 类访问数据

3. 在程序的AndroidManifest.xml文件中声明。

继承ContentProvider需要实现这六个方法:

query()
insert()
update()
delete()
getType()
onCreate()

注意:因为ContentProvider可能被不同的进程和线程调用,所以这些方法必须是线程安全的。

package com.example.android.notepad;

import android.net.Uri;
import android.provider.BaseColumns;

/**
 * Convenience definitions for NotePadProvider
 * Content Provider的基础类,该类定义了每一列的名字,内容Uri,默认排序方式等常量
 */
public final class NotePad {
    public static final String AUTHORITY = "com.google.provider.NotePad";

    // This class cannot be instantiated
    private NotePad() {}
    
    /**
     * Notes table
     * BaseColumns定义了两个基本字段_ID(每一行的ID)和_COUNT(每个目录下的行数)
     */
    public static final class Notes implements BaseColumns {
        // This class cannot be instantiated
        private Notes() {}

        /**
         * The content:// style URL for this table
         */
        public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/notes");

        /**
         * The MIME type of {@link #CONTENT_URI} providing a directory of notes.
         * 定义新的MIME类型对应notes的目录
         */
        public static final String CONTENT_TYPE = "vnd.android.cursor.dir/vnd.google.note";

        /**
         * The MIME type of a {@link #CONTENT_URI} sub-directory of a single note.
         * 定义新的MIME类型对应一个note
         */
        public static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/vnd.google.note";

        /**
         * The default sort order for this table
         */
        public static final String DEFAULT_SORT_ORDER = "modified DESC";

        /**
         * The title of the note
         * <P>Type: TEXT</P>
         */
        public static final String TITLE = "title";

        /**
         * The note itself
         * <P>Type: TEXT</P>
         */
        public static final String NOTE = "note";

        /**
         * The timestamp for when the note was created
         * <P>Type: INTEGER (long from System.curentTimeMillis())</P>
         */
        public static final String CREATED_DATE = "created";

        /**
         * The timestamp for when the note was last modified
         * <P>Type: INTEGER (long from System.curentTimeMillis())</P>
         */
        public static final String MODIFIED_DATE = "modified";
    }
}

 

package com.example.android.notepad;

import com.example.android.notepad.NotePad.Notes;

import android.content.ContentProvider;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.Context;
import android.content.UriMatcher;
import android.content.res.Resources;
import android.database.Cursor;
import android.database.SQLException;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import android.database.sqlite.SQLiteQueryBuilder;
import android.net.Uri;
import android.text.TextUtils;
import android.util.Log;

import java.util.HashMap;

/**
 * Provides access to a database of notes. Each note has a title, the note
 * itself, a creation date and a modified data.
 */
public class NotePadProvider extends ContentProvider {

    private static final String TAG = "NotePadProvider";
		//数据库名
    private static final String DATABASE_NAME = "note_pad.db";
    private static final int DATABASE_VERSION = 2;
    //表名
    private static final String NOTES_TABLE_NAME = "notes";

    private static HashMap<String, String> sNotesProjectionMap;

    private static final int NOTES = 1;
    private static final int NOTE_ID = 2;

    private static final UriMatcher sUriMatcher;

    /**
     * This class helps open, create, and upgrade the database file.
     */
    private static class DatabaseHelper extends SQLiteOpenHelper {

        DatabaseHelper(Context context) {
            super(context, DATABASE_NAME, null, DATABASE_VERSION);
        }

        @Override
        public void onCreate(SQLiteDatabase db) {
            db.execSQL("CREATE TABLE " + NOTES_TABLE_NAME + " ("
                    + Notes._ID + " INTEGER PRIMARY KEY,"
                    + Notes.TITLE + " TEXT,"
                    + Notes.NOTE + " TEXT,"
                    + Notes.CREATED_DATE + " INTEGER,"
                    + Notes.MODIFIED_DATE + " INTEGER"
                    + ");");
        }

        @Override
        public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
            Log.w(TAG, "Upgrading database from version " + oldVersion + " to "
                    + newVersion + ", which will destroy all old data");
            db.execSQL("DROP TABLE IF EXISTS notes");
            onCreate(db);
        }
    }

    private DatabaseHelper mOpenHelper;

    @Override
    public boolean onCreate() {
        mOpenHelper = new DatabaseHelper(getContext());
        return true;
    }

    @Override
    public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs,
            String sortOrder) {
        SQLiteQueryBuilder qb = new SQLiteQueryBuilder();

        switch (sUriMatcher.match(uri)) {
        case NOTES:
            qb.setTables(NOTES_TABLE_NAME);
            qb.setProjectionMap(sNotesProjectionMap);
            break;

        case NOTE_ID:
            qb.setTables(NOTES_TABLE_NAME);
            qb.setProjectionMap(sNotesProjectionMap);
            qb.appendWhere(Notes._ID + "=" + uri.getPathSegments().get(1));
            break;

        default:
            throw new IllegalArgumentException("Unknown URI " + uri);
        }

        // If no sort order is specified use the default
        String orderBy;
        if (TextUtils.isEmpty(sortOrder)) {
            orderBy = NotePad.Notes.DEFAULT_SORT_ORDER;
        } else {
            orderBy = sortOrder;
        }

        // Get the database and run the query
        SQLiteDatabase db = mOpenHelper.getReadableDatabase();
        Cursor c = qb.query(db, projection, selection, selectionArgs, null, null, orderBy);

        // Tell the cursor what uri to watch, so it knows when its source data changes
        c.setNotificationUri(getContext().getContentResolver(), uri);
        return c;
    }

    @Override
    //如果有自定义类型,必须实现该方法
    public String getType(Uri uri) {
        switch (sUriMatcher.match(uri)) {
        case NOTES:
            return Notes.CONTENT_TYPE;

        case NOTE_ID:
            return Notes.CONTENT_ITEM_TYPE;

        default:
            throw new IllegalArgumentException("Unknown URI " + uri);
        }
    }

    @Override
    public Uri insert(Uri uri, ContentValues initialValues) {
        // Validate the requested uri
        if (sUriMatcher.match(uri) != NOTES) {
            throw new IllegalArgumentException("Unknown URI " + uri);
        }

        ContentValues values;
        if (initialValues != null) {
            values = new ContentValues(initialValues);
        } else {
            values = new ContentValues();
        }

        Long now = Long.valueOf(System.currentTimeMillis());

        // Make sure that the fields are all set
        if (values.containsKey(NotePad.Notes.CREATED_DATE) == false) {
            values.put(NotePad.Notes.CREATED_DATE, now);
        }

        if (values.containsKey(NotePad.Notes.MODIFIED_DATE) == false) {
            values.put(NotePad.Notes.MODIFIED_DATE, now);
        }

        if (values.containsKey(NotePad.Notes.TITLE) == false) {
            Resources r = Resources.getSystem();
            values.put(NotePad.Notes.TITLE, r.getString(android.R.string.untitled));
        }

        if (values.containsKey(NotePad.Notes.NOTE) == false) {
            values.put(NotePad.Notes.NOTE, "");
        }

        SQLiteDatabase db = mOpenHelper.getWritableDatabase();
        long rowId = db.insert(NOTES_TABLE_NAME, Notes.NOTE, values);
        if (rowId > 0) {
            Uri noteUri = ContentUris.withAppendedId(NotePad.Notes.CONTENT_URI, rowId);
            getContext().getContentResolver().notifyChange(noteUri, null);
            return noteUri;
        }

        throw new SQLException("Failed to insert row into " + uri);
    }

    @Override
    public int delete(Uri uri, String where, String[] whereArgs) {
        SQLiteDatabase db = mOpenHelper.getWritableDatabase();
        int count;
        switch (sUriMatcher.match(uri)) {
        case NOTES:
            count = db.delete(NOTES_TABLE_NAME, where, whereArgs);
            break;

        case NOTE_ID:
            String noteId = uri.getPathSegments().get(1);
            count = db.delete(NOTES_TABLE_NAME, Notes._ID + "=" + noteId
                    + (!TextUtils.isEmpty(where) ? " AND (" + where + ')' : ""), whereArgs);
            break;

        default:
            throw new IllegalArgumentException("Unknown URI " + uri);
        }

        getContext().getContentResolver().notifyChange(uri, null);
        return count;
    }

    @Override
    public int update(Uri uri, ContentValues values, String where, String[] whereArgs) {
        SQLiteDatabase db = mOpenHelper.getWritableDatabase();
        int count;
        switch (sUriMatcher.match(uri)) {
        case NOTES:
            count = db.update(NOTES_TABLE_NAME, values, where, whereArgs);
            break;

        case NOTE_ID:
            String noteId = uri.getPathSegments().get(1);
            count = db.update(NOTES_TABLE_NAME, values, Notes._ID + "=" + noteId
                    + (!TextUtils.isEmpty(where) ? " AND (" + where + ')' : ""), whereArgs);
            break;

        default:
            throw new IllegalArgumentException("Unknown URI " + uri);
        }

        getContext().getContentResolver().notifyChange(uri, null);
        return count;
    }

    static {
        sUriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
        sUriMatcher.addURI(NotePad.AUTHORITY, "notes", NOTES);
        sUriMatcher.addURI(NotePad.AUTHORITY, "notes/#", NOTE_ID);

        sNotesProjectionMap = new HashMap<String, String>();
        sNotesProjectionMap.put(Notes._ID, Notes._ID);
        sNotesProjectionMap.put(Notes.TITLE, Notes.TITLE);
        sNotesProjectionMap.put(Notes.NOTE, Notes.NOTE);
        sNotesProjectionMap.put(Notes.CREATED_DATE, Notes.CREATED_DATE);
        sNotesProjectionMap.put(Notes.MODIFIED_DATE, Notes.MODIFIED_DATE);
    }
}

 

分享到:
评论

相关推荐

    Notepad++文本编辑器学习手册

    Notepad++文本编辑器的学习手册

    notepad++主题文件

    notepad++主题文件notepad++主题文件notepad++主题文件notepad++主题文件notepad++主题文件notepad++主题文件notepad++主题文件notepad++主题文件notepad++主题文件notepad++主题文件notepad++主题文件notepad++主题...

    Notepad++安装包

    Notepad++中文版是一款非常有特色的编辑器,是开源软件,可以免费使用。 Notepad++中文版是 Windows操作系统下的一套文本编辑器(软件版权许可证: GPL)。Notepad++有完整的中文化接口及支持多国语言编写的功能(UTF8...

    Notepad++ 解压即用

    14. **多视图编辑**:在单个Notepad++实例中,可以同时打开两个或更多文件进行并排比较和编辑。 15. **书签功能**:用户可以在代码中设置书签,便于快速跳转到特定位置。 总的来说,Notepad++因其轻量级、高效和...

    notepad3最新版 (2022-11-19) 完美替代notepad++

    Notepad3是一款文本编辑器,作为Notepad++的优秀替代品,它在2022年11月19日发布了最新的版本。这个版本旨在提供更高效、更便捷的文本编辑体验,尤其针对程序员和高级用户,具备了丰富的代码高亮、自动完成以及语法...

    Notepad Verilog 插件nppVerilog

    Notepad++是一款非常受欢迎的文本编辑器,尤其在编程领域,因其轻量级、可自定义及支持多种语言而受到广大程序员的喜爱。针对VERILOG这种硬件描述语言,Notepad++提供了一些插件来增强其功能,其中之一就是...

    NotePad++安装包

    1. 获取安装包:NotePad++的最新版本为8.5.3,可以从官方网站下载到对应操作系统的安装文件,例如“npp.8.5.3.Installer.x64.exe”是适用于64位Windows系统的安装程序。 2. 执行安装:双击下载的安装文件,启动安装...

    notepad++ V8.3.3源码

    通过学习Notepad++ V8.3.3的源码,不仅可以提升我们的C++编程技巧,还能深入了解文本编辑器的设计原理,为开发自己的文本处理工具或增强现有项目提供灵感。同时,源码阅读也是锻炼编程思维、提高代码质量的有效途径...

    notepad

    notepad

    Notepad-plus-plus-V8.6.4

    《深入理解Notepad++:基于Notepad-plus-plus-V8.6.4的探索》 Notepad++是一款广受欢迎的开源文本编辑器,以其轻量级、高效和强大的编程功能著称。版本V8.6.4是其历史版本中的一个重要里程碑,提供了许多改进和新...

    Qt小实例-记事本notepad

    **Qt小实例 - 记事本Notepad** 在IT领域,Qt是一个强大的跨平台应用程序开发框架,由The Qt Company提供,广泛应用于桌面...Qt小实例如记事本Notepad是学习和实践的最佳入口,它将帮助你建立起对Qt开发的信心和技巧。

    notepad ++打开双窗口的方法

    - **菜单栏**:点击菜单栏上的“窗口”&gt;“新文档”,或者按`Ctrl+N`,在当前Notepad++实例中新建一个空白文档。接着,通过“文件”&gt;“打开”加载你需要的文件。 - **复制窗口**:如果你已经打开了一个文件,可以...

    Notepad++ notepad

    Notepad . it is a useful software

    notepad安装包及插件

    标题 "notepad安装包及插件" 涉及到的是一个知名文本编辑器Notepad++的相关内容。Notepad++是一款免费且开源的代码编辑器,尤其受到程序员和IT专业人员的喜爱,因为它支持多种编程语言,并具有丰富的自定义功能。 ...

    notepad的PluginManager

    3. 关闭正在运行的所有Notepad++实例,以确保安装过程中不会出现冲突。 4. 找到Notepad++的安装目录,通常位于`C:\Program Files (x86)\Notepad++`或`C:\Program Files\Notepad++`(根据你的系统架构)。 5. 在该...

    Notepad++.zip

    不仅如此,Notepad++还是一款开源软件,遵循GPL协议,这意味着它的源代码可供任何人查看、学习和改进。这促进了社区对它的持续优化和更新,确保了软件的稳定性和功能性。 总而言之,Notepad++是一款功能强大且高度...

    Notepad++7.9安装包

    Notepad++是一款非常受欢迎的免费源代码编辑器,尤其在编程领域中被广泛使用。它基于Windows操作系统,支持多种编程语言,并且具有丰富的自定义功能。Notepad++ 7.9版本是该软件的一个更新迭代,带来了许多改进和新...

    notepad++ 多开补丁

    标题提到的“多开补丁”是指允许用户在同一时间运行多个Notepad++实例的补丁文件。 Notepad++的默认设置通常只允许一次启动一个实例,这在某些情况下可能限制了用户同时处理多个文件或项目的能力。多开补丁就是为了...

    Notepad++免安装版

    Notepad++是一款非常受...无论是在学习编程、编写脚本还是日常的文本处理工作中,Notepad++都能够提供高效、便捷的编辑体验。其免安装特性更是方便了用户在不同设备间灵活切换,满足了现代IT工作场景中的多样化需求。

Global site tag (gtag.js) - Google Analytics