ContentProvider 可以用来原生读写 Android 自带的数据库 SQLite。

  1. 使用 Studio 创建一个 ContentProvider, 名字叫 TestContentProvider 。

    AndroidManifest.xml 文件:

    <manifest
    ...>
    <application
    ...>
    <provider
    android:name=".TestContentProvider"
    android:authorities="cn.wx2020.contentprovidertest.database"
    android:enabled="true"
    android:exported="false"/>
    ...
    </application>
    </manifest>

    其中,authorities 是 Uri 结构中的一部分,类似于 Url 中的域名部分,其唯一识别一个 ContentProvider

  2. 由于 ContentProvider 是抽象类,按照继承抽象类的要求,需要实现以下几个方法:

    @Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
// Implement this to handle requests to delete one or more rows.
throw new UnsupportedOperationException("Not yet implemented");
} @Override
public String getType(Uri uri) {
// TODO: Implement this to handle requests for the MIME type of the data
// at the given URI.
throw new UnsupportedOperationException("Not yet implemented");
} @Override
public Uri insert(Uri uri, ContentValues values) {
// TODO: Implement this to handle requests to insert a new row.
throw new UnsupportedOperationException("Not yet implemented");
} @Override
public boolean onCreate() {
// TODO: Implement this to initialize your content provider on startup.
return false;
} @Override
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder) {
// TODO: Implement this to handle query requests from clients.
throw new UnsupportedOperationException("Not yet implemented");
} @Override
public int update(Uri uri, ContentValues values, String selection,
String[] selectionArgs) {
// TODO: Implement this to handle requests to update one or more rows.
throw new UnsupportedOperationException("Not yet implemented");
}
  1. 本文实际只实现了 onCreate()insert()query() 这三个方法。
  • onCreate():负责初始化数据库帮助类 SQLiteOpenHelper,通过此类可以进行数据库创建、升级等操作。
      private TestSQLiteOpenHelper mOpenHelper;
    ... @Override
    public boolean onCreate() {
    mOpenHelper = new TestSQLiteOpenHelper(getContext());
    return false;
    }

    其中,

    • TestSQLiteOpenHelper 为 继承 SQLiteOpenHelper 后自定义的实现类,将在后文给出说明。
    • 此方法的返回值意义不大,可以忽略。
  • insert():负责在最内部一层将传入的数据插入数据库中。
      public static final String AUTHORITY = "cn.wx2020.contentprovidertest.database";
    
      private static final UriMatcher MATCHER;
    
      static {
    MATCHER = new UriMatcher(UriMatcher.NO_MATCH);
    MATCHER.addURI(AUTHORITY, "user", 1);
    } private String uriMatcher(Uri uri) {
    switch (MATCHER.match(uri)) {
    case 1:
    return "user";
    default:
    return "";
    }
    }
    ... @Override
    public Uri insert(Uri uri, ContentValues values) {
    String tableName = uriMatcher(uri);
    if (TextUtils.isEmpty(tableName)) {
    return null;
    }
    SQLiteDatabase db = mOpenHelper.getWritableDatabase();
    long rowName = db.insert(tableName, null, values);
    if (rowName >= 0) {
    Uri returnUri = ContentUris.withAppendedId(uri, rowName);
    getContext().getContentResolver().notifyChange(uri, null);
    return returnUri;
    }
    return null;
    }

    其中,

    • 方法 uriMatcher() 使用静态的 UriMatcher ,在静态块中初始化、添加匹配当前 ContentProviderauthorities、数据库表名 和 与数据库表名绑定的返回码。uriMatcher() 方法中,使用 MATCHER.match(uri) 拿到返回码后,根据返回码再返回与其匹配的表名。
    • 使用 SQLiteOpenHelper 的对象调用 getWritableDatabase() 方法拿到可写的数据库后,对数据库 db 调用 insert() 方法插入传入的属性值,属性值由 ContentValues 代理生成。插入成功后,insert() 函数会返回一个 rowId ;如果插入失败,rowId-1;若其他情况则插入成功。
    • 插入成功时,返回一个 Uri 对象,上层根据这个 Uri 对象,判断是否插入成功。
  • query():负责查询数据库中的数据。
      @Override
    public Cursor query(Uri uri, String[] projection, String selection,
    String[] selectionArgs, String sortOrder) {
    String tableName = uriMatcher(uri);
    if (TextUtils.isEmpty(tableName)) {
    return null;
    } Cursor cursor = null;
    try {
    SQLiteDatabase db = mOpenHelper.getReadableDatabase();
    SQLiteQueryBuilder builder = new SQLiteQueryBuilder();
    builder.setTables(tableName);
    cursor = builder.query(db, projection, selection, selectionArgs, null, null, sortOrder);
    } catch (SQLException e) {
    Log.e(TAG, "query SQLiteException");
    } if (cursor != null) {
    cursor.setNotificationUri(getContext().getContentResolver(), uri);
    } else {
    Log.e(TAG, "cursor = null");
    }
    return cursor;
    }

    其中,

    • 拿到可读的数据库后,使用 SQLiteQueryBuilder 的对象 builder 构造查询。使用 builder.setTables() 设定要查询的数据库表名。使用 builder.query() 执行查询,按照查询条件依次输入7个参数,其中第一个参数为数据库对象 db
    • 查询得到的游标 Cursor 对象如果不为 null, 说明查询成功,否则则为查询失败。返回 Cursor 对象给调用方。
  1. 上文中还提到了一个类 SQLiteOpenHelper ,其内部有两个方法:onCreate() 在数据库创建后调用;onUpgrade() 升级数据库版本时使用。
  • onCreate()
      @Override
    public void onCreate(SQLiteDatabase db) {
    createDatabaseTable(db);
    } private void createDatabaseTable(SQLiteDatabase db) {
    String userSQL = "CREATE TABLE IF NOT EXISTS user (" +
    " _id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT," +
    " user_name TEXT NOT NULL," +
    " age TEXT," +
    " phone_number INTEGER" +
    ");";
    db.execSQL(userSQL);
    }

    此方法中,要使用 SQL 语句在指定数据库创建表。使用 String 构造好建表语句后,调用数据库 dbexecSQL() 执行 SQL 语句。

  1. 除了以上两个类,还可以在实际调用 ContentProvider 之前,在数据库表的维度中再抽象出一层 UserTable,用于外界进行实际查询时调用。由于 ContentProvider 只实现了 insert()query() 方法,相对应地,UserTable 也实现两个方法 insert()queryAll()
  • insert():用于实际调用 ContentProvider,插入数据。
      private static Uri getContextUri() {
    return TestContentProvider.URI_TEST_USER;
    } public static Uri insert(Context context, User user) {
    if (TextUtils.isEmpty(user.getUserName())) {
    return null;
    }
    ContentValues contentValues = new ContentValues();
    contentValues.put("user_name", user.getUserName());
    contentValues.put("age", user.getAge());
    contentValues.put("phone_number", user.getPhoneNumber());
    Uri uri = null;
    try {
    uri = context.getContentResolver().insert(getContextUri(), contentValues);
    } catch (SQLException e) {
    Log.e(TAG, "insert SQLiteException");
    }
    if (uri == null) {
    Log.e(TAG, "insert uri is null");
    }
    return uri;
    }

    其中,

    • 静态方法getContextUri() 用于返回与数据库当前表对应的 Uri ,操作 ContentProvider 也就是通过 Uri 操作的。
    • 方法参数中,实体类 User 的对象中的属性值正是插入数据库时所使用的字段值,不过需要将实体类对象转换为 ContentValues 对象后,使其作为 ContentResolver 类对象的 insert() 方法参数之一进行插入。
  • queryAll(): 用于查询数据库表 user 中的所有数据。
      public static ArrayList<User> queryAll(Context context) {
    ArrayList<User> result = new ArrayList<>();
    Cursor cursor = null;
    try {
    cursor = context.getContentResolver().query(getContextUri(), null, null, null, null);
    while (cursor != null && cursor.moveToNext()) {
    User user = new User();
    user.setId(cursor.getInt(cursor.getColumnIndexOrThrow("_id")));
    user.setUserName(cursor.getString(cursor.getColumnIndexOrThrow("user_name")));
    user.setAge(cursor.getInt(cursor.getColumnIndexOrThrow("age")));
    user.setPhoneNumber(cursor.getLong(cursor.getColumnIndexOrThrow("phone_number")));
    result.add(user);
    }
    } catch (SQLException e) {
    Log.e(TAG, "queryAll SQLiteException");
    } finally {
    if (cursor != null && !cursor.isClosed()) {
    cursor.close();
    }
    }
    return result;
    }

    其中,

    • 使用 ContentResolver 类对象的 query() 方法,除了 Uri 以外不传任何参数,这样获取的 cursor 就是

      在整个链上连续且整体代表全部表数据的 cursor 。
    • 使用 cursor != null && cursor.moveToNext() 作为循环条件,判断链上是否有下一个数据。若没有下一个数据可以终止循环。
    • 每一行的数据是从 cursor 中取得的,其中 cursor 要先通过 getColumnIndexOrThrow() 传入字段名,拿到代表这一字段的 index ,再通过对 index 调用 cursor.getInt()cursor.getString() 拿到这一 index 代表的字段的 INT 或者 TEXT 值。
    • 每拿到一个 cursor 对应的数据后,将数据加入 ArrayList 中,最后将 ArrayList 对象作为整体返回。

部分源码

  1. TestContentProvider 源码:
package cn.wx2020.contentprovidertest;

import android.content.ContentProvider;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.UriMatcher;
import android.database.Cursor;
import android.database.SQLException;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteQueryBuilder;
import android.net.Uri;
import android.text.TextUtils;
import android.util.Log; import java.util.Optional; public class TestContentProvider extends ContentProvider { public static final String TAG = "TestContentProvider"; public static final String AUTHORITY = "cn.wx2020.contentprovidertest.database"; public static final Uri URI_TEST_USER = Uri.parse("content://" + AUTHORITY + '/' + "user"); private static final UriMatcher MATCHER; static {
MATCHER = new UriMatcher(UriMatcher.NO_MATCH);
MATCHER.addURI(AUTHORITY, "user", 1);
} private String uriMatcher(Uri uri) {
switch (MATCHER.match(uri)) {
case 1:
return "user";
default:
return "";
}
} private TestSQLiteOpenHelper mOpenHelper; public TestContentProvider() {
} @Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
// Implement this to handle requests to delete one or more rows.
throw new UnsupportedOperationException("Not yet implemented");
} @Override
public String getType(Uri uri) {
// TODO: Implement this to handle requests for the MIME type of the data
// at the given URI.
throw new UnsupportedOperationException("Not yet implemented"); } @Override
public Uri insert(Uri uri, ContentValues values) {
String tableName = uriMatcher(uri);
if (TextUtils.isEmpty(tableName)) {
return null;
}
SQLiteDatabase db = mOpenHelper.getWritableDatabase();
long rowName = db.insert(tableName, null, values);
if (rowName >= 0) {
Uri returnUri = ContentUris.withAppendedId(uri, rowName);
getContext().getContentResolver().notifyChange(uri, null);
return returnUri;
}
return null;
} @Override
public boolean onCreate() {
mOpenHelper = new TestSQLiteOpenHelper(getContext());
return false;
} @Override
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder) {
String tableName = uriMatcher(uri);
if (TextUtils.isEmpty(tableName)) {
return null;
} Cursor cursor = null;
try {
SQLiteDatabase db = mOpenHelper.getReadableDatabase();
SQLiteQueryBuilder builder = new SQLiteQueryBuilder();
builder.setTables(tableName);
cursor = builder.query(db, projection, selection, selectionArgs, null, null, sortOrder);
} catch (SQLException e) {
Log.e(TAG, "query SQLiteException");
} if (cursor != null) {
cursor.setNotificationUri(getContext().getContentResolver(), uri);
} else {
Log.e(TAG, "cursor = null");
}
return cursor;
} @Override
public int update(Uri uri, ContentValues values, String selection,
String[] selectionArgs) {
// TODO: Implement this to handle requests to update one or more rows.
throw new UnsupportedOperationException("Not yet implemented");
}
}
  1. TestSqLiteOpenHelper 源码:
package cn.wx2020.contentprovidertest;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper; public class TestSQLiteOpenHelper extends SQLiteOpenHelper { private static final String DB_NAME = "test.db"; private static final int DB_VERSION = 1; public TestSQLiteOpenHelper(Context context) {
super(context, DB_NAME, null, DB_VERSION);
} @Override
public void onCreate(SQLiteDatabase db) {
createDatabaseTable(db);
} private void createDatabaseTable(SQLiteDatabase db) {
String userSQL = "CREATE TABLE IF NOT EXISTS user (" +
" _id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT," +
" user_name TEXT NOT NULL," +
" age TEXT," +
" phone_number INTEGER" +
");";
db.execSQL(userSQL);
} @Override
public void onUpgrade(SQLiteDatabase sqLiteDatabase, int i, int i1) { }
}
  1. UserTable源码:
package cn.wx2020.contentprovidertest;

import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.database.SQLException;
import android.net.Uri;
import android.text.TextUtils;
import android.util.Log; import java.util.ArrayList; public class UserTable {
private static final String TAG = "UserTable"; private static Uri getContextUri() {
return TestContentProvider.URI_TEST_USER;
} public static Uri insert(Context context, User user) {
if (TextUtils.isEmpty(user.getUserName())) {
return null;
}
ContentValues contentValues = new ContentValues();
contentValues.put("user_name", user.getUserName());
contentValues.put("age", user.getAge());
contentValues.put("phone_number", user.getPhoneNumber());
Uri uri = null;
try {
uri = context.getContentResolver().insert(getContextUri(), contentValues);
} catch (SQLException e) {
Log.e(TAG, "insert SQLiteException");
}
if (uri == null) {
Log.e(TAG, "insert uri is null");
}
return uri;
} public static ArrayList<User> queryAll(Context context) {
ArrayList<User> result = new ArrayList<>();
Cursor cursor = null;
try {
cursor = context.getContentResolver().query(getContextUri(), null, null, null, null);
while (cursor != null && cursor.moveToNext()) {
User user = new User();
user.setId(cursor.getInt(cursor.getColumnIndexOrThrow("_id")));
user.setUserName(cursor.getString(cursor.getColumnIndexOrThrow("user_name")));
user.setAge(cursor.getInt(cursor.getColumnIndexOrThrow("age")));
user.setPhoneNumber(cursor.getLong(cursor.getColumnIndexOrThrow("phone_number")));
result.add(user);
}
} catch (SQLException e) {
Log.e(TAG, "queryAll SQLiteException");
} finally {
if (cursor != null && !cursor.isClosed()) {
cursor.close();
}
}
return result;
}
}
  1. MainActivity 源码:
        Button queryAllButton = findViewById(R.id.button_1);
TextView textView = findViewById(R.id.text_1);
queryAllButton.setOnClickListener(v -> {
ArrayList<User> users = UserTable.queryAll(this);
if (users.isEmpty()) {
Log.e(TAG, "users is empty");
Toast.makeText(this, "没有数据", Toast.LENGTH_SHORT).show();
return;
}
StringBuilder sb = new StringBuilder();
for (User user: users) {
sb.append(user.toString()).append("\n");
}
textView.setText(sb.toString());
}); Button insertButton = findViewById(R.id.button_2);
insertButton.setOnClickListener(v -> {
User user = new User();
int userBack = (int) (Math.random() * 99 + 1);
user.setUserName("wx2020" + userBack);
user.setPhoneNumber(110);
user.setAge((int) (Math.random() * 30 + 23));
Uri insertUri = UserTable.insert(this, user);
if (insertUri == null) {
Toast.makeText(this, "插入失败", Toast.LENGTH_SHORT).show();
return;
}
Log.d(TAG, "insertUri = " + insertUri);
Toast.makeText(this, "插入成功", Toast.LENGTH_SHORT).show();
});

Android 使用 ContentProvider 简单操作数据库的更多相关文章

  1. Spring_boot简单操作数据库

    Spring_boot搭配Spring Data JPA简单操作数据库 spring boot 配置文件可以使用yml文件,默认spring boot 会加载resources目录的下的applica ...

  2. android: SQLite使用 SQL 操作数据库

    虽然 Android 已经给我们提供了很多非常方便的 API 用于操作数据库,不过总会有一些 人不习惯去使用这些辅助性的方法,而是更加青睐于直接使用 SQL 来操作数据库.这种人 一般都是属于 SQL ...

  3. php 简单操作数据库

    <?php header("content-type:text/html;charset=utf-8"); /*//造一个连接 $connect = @mysql_conne ...

  4. pymysql 简单操作数据库

    #!/usr/bin/env python #-*- coding:utf-8 -*- # author:leo # datetime:2019/4/24 15:22 # software: PyCh ...

  5. Android使用命令行操作数据库

    所有的应用程序本地文件都存放在/data/data/目录下 C:\Users\nicole>adb shell * daemon not running. starting it now on ...

  6. mybatis_02简单操作数据库

    模糊查询用户信息 <!-- [${}]:表示拼接SQL字符串 [${value}]:表示要拼接的是简单类型参数. 注意: 1.如果参数为简单类型时,${}里面的参数名称必须为value 2.${ ...

  7. SQLiteDatabase里面的简单操作数据库的方法

    1.使用insert方法插入记录SQLiteDatabase的insert方法的签名为long insert(String table,String nullColumnHack,ContentVal ...

  8. spring框架整合hibernate框架简单操作数据库

    1.配置文件: <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http:/ ...

  9. pyqt5最简单操作数据库

    要先安一个包才能使用QtSql通过新立得安装 import PyQt5.QtSql as sql db=sql.QSqlDatabase.addDatabase('QMYSQL') db.setDat ...

  10. spring-boot-route(七)整合jdbcTemplate操作数据库

    在一部分内容中,我们学习了Restful接口的编写,及接口文档的生成.我们需要将接口数据进行持久化存储,这一部分我们主要学习几种持久化框架将数据进行存储.本部分内容中,我们都将使用mysql为例来做为 ...

随机推荐

  1. Ceres简单应用-求解(Powell's Function)鲍威尔函数最小值

    Ceres 求解 Powell's function 的最小化 \(\quad\)现在考虑一个稍微复杂一点的例子-鲍威尔函数的最小化. \(\quad{}\) \(x=[x_1,x_2,x_3,x_4 ...

  2. Dirty-Pipe Linux内核提权漏洞(CVE-2022-0847)

    前言: 划水一波,哈哈,以后复现漏洞不再直接傻瓜无脑的走流程了,首先码字写加构思比较麻烦且写的不多还效率不高,现在就是当做见到了一个漏洞,在此记录一下这个漏洞,包括其来源,简单的描述,适用范围,以及其 ...

  3. 产品代码都给你看了,可别再说不会DDD(三):战略设计

    这是一个讲解DDD落地的文章系列,作者是<实现领域驱动设计>的译者滕云.本文章系列以一个真实的并已成功上线的软件项目--码如云(https://www.mryqr.com)为例,系统性地讲 ...

  4. Jitpack发布Android库带文档和源码

    原文地址: Jitpack发布Android库带文档和源码 - Stars-One的杂货小窝 忽然发现自己发布的xAndroidUtil库 写代码的时候看方法注释都看不到,研究了下如何让Jitpack ...

  5. O2OA(翱途)开发平台 V8.1正式发布

    尊敬的O2OA(翱途)平台合作伙伴.用户以及亲爱的开发小伙伴们,平台 V8.1版本已正式发布.正值8月的最后一周,我们以更安全.更高效.更好用的崭新面貌迎接9月的到来. O2OA开发平台v8.1版本更 ...

  6. 本地项目上传到Git仓库

    1. 进入项目主目录,打开Git Bash,执行以下命令,将项目变为一个git管理的项目: $ git init 执行成功后,会在项目根目录生成一个.git的文件夹. 可以执行以下命令查看项目状态: ...

  7. 「joisc2016 - D3T2」回転寿司

    题意大概是这样,「每次操作选出区间中的一个 LIS(strictly),满足其开端是极靠近左端点且大于 \(A\) 的位置,答案即这个 LIS 的末尾,做一个轮换后弹出序列末端」. 首先做几个观察. ...

  8. 算法——AcWing算法提高课中代码和题解

    文章目录 第一章 动态规划 (完成情况:64/68) 数字三角形模型 最长上升子序列模型 背包模型 状态机模型 状态压缩DP 区间DP 树形DP 数位DP 单调队列优化DP 斜率优化DP 第二章 搜索 ...

  9. oracle数据库性能监控常用sql

    因执行时间较长建议使用plsql等第三方工具执行 --1.监控sga内存分配信息select * from v$sgainfo;--2.监控每个用户的磁盘io及io命中率select v$sess_i ...

  10. sql分组后排序计算

    用法:RANK() OVER(PARTITION BY 分组字段 ORDER BY 排序字段 ) 例子:要得到n4列 ---创建测试数据create table tb(n1 varchar2(40) ...