深入分析 Android ContentProvider (五)

文章目录

    • 深入分析 Android ContentProvider (五)
    • ContentProvider 的性能优化和实践案例
      • 1. 性能优化技巧
        • 1.1. 数据库索引优化
          • 示例:添加索引
        • 1.2. 批量操作与事务管理
          • 示例:批量插入操作
        • 1.3. 使用异步操作
          • 示例:使用 AsyncTask 进行异步查询
        • 1.4. 缓存机制
          • 示例:使用 LruCache 进行内存缓存
        • 1.5. 使用 Loader 进行异步加载
          • 示例:使用 CursorLoader
      • 2. 实践案例
        • 2.1 案例一:消息应用的数据同步
          • 消息 ContentProvider 实现
        • 2.2 案例二:音乐播放器的媒体库管理
          • 媒体库 ContentProvider 实现
      • 3. 总结

深入分析 Android ContentProvider (五)

ContentProvider 的性能优化和实践案例

在实践中,合理的性能优化可以显著提升 ContentProvider 的效率和用户体验。以下是一些具体的性能优化技巧和实际案例,以便更好地理解和应用 ContentProvider。

1. 性能优化技巧

1.1. 数据库索引优化

在频繁进行查询操作的字段上添加索引,可以显著提高查询性能。数据库索引可以加快数据检索的速度,尤其是在大型数据集上。

示例:添加索引

在创建数据库表时,可以通过 SQL 语句为某些列创建索引:

private static final String CREATE_TABLE ="CREATE TABLE " + TABLE_NAME + " (" +COLUMN_ID + " INTEGER PRIMARY KEY AUTOINCREMENT, " +COLUMN_NAME + " TEXT NOT NULL);";private static final String CREATE_INDEX ="CREATE INDEX index_name ON " + TABLE_NAME + " (" + COLUMN_NAME + ");";@Override
public void onCreate(SQLiteDatabase db) {db.execSQL(CREATE_TABLE);db.execSQL(CREATE_INDEX);
}
1.2. 批量操作与事务管理

在执行大批量的数据插入、更新或删除操作时,使用事务可以减少数据库锁的开销,并提高操作的整体性能。

示例:批量插入操作
public void bulkInsertData(List<ContentValues> valuesList) {SQLiteDatabase db = dbHelper.getWritableDatabase();db.beginTransaction();try {for (ContentValues values : valuesList) {db.insertOrThrow(TABLE_NAME, null, values);}db.setTransactionSuccessful();} finally {db.endTransaction();}
}
1.3. 使用异步操作

避免在主线程中进行数据库操作,使用 AsyncTaskLoaderRxJava 等异步框架进行数据操作,确保应用的 UI 流畅性。

示例:使用 AsyncTask 进行异步查询
private class QueryTask extends AsyncTask<Void, Void, Cursor> {@Overrideprotected Cursor doInBackground(Void... voids) {Uri uri = Uri.parse("content://com.example.provider/examples");return getContentResolver().query(uri, null, null, null, "name ASC");}@Overrideprotected void onPostExecute(Cursor cursor) {if (cursor != null) {// 处理查询结果cursor.close();}}
}
1.4. 缓存机制

在数据访问频繁的场景中,使用缓存机制可以显著提高性能。可以选择内存缓存(如 LruCache)或磁盘缓存来缓存常用数据,减少数据库查询的次数。

示例:使用 LruCache 进行内存缓存
private LruCache<String, Bitmap> memoryCache;public void initCache() {final int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024);final int cacheSize = maxMemory / 8;memoryCache = new LruCache<>(cacheSize);
}public void addBitmapToCache(String key, Bitmap bitmap) {if (getBitmapFromCache(key) == null) {memoryCache.put(key, bitmap);}
}public Bitmap getBitmapFromCache(String key) {return memoryCache.get(key);
}
1.5. 使用 Loader 进行异步加载

Loader 可以在异步线程中加载数据,避免主线程阻塞,并在数据加载完成时自动更新 UI。

示例:使用 CursorLoader
public class ExampleActivity extends AppCompatActivity implements LoaderManager.LoaderCallbacks<Cursor> {private static final int LOADER_ID = 1;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_example);getSupportLoaderManager().initLoader(LOADER_ID, null, this);}@NonNull@Overridepublic Loader<Cursor> onCreateLoader(int id, @Nullable Bundle args) {Uri uri = Uri.parse("content://com.example.provider/examples");return new CursorLoader(this, uri, null, null, null, "name ASC");}@Overridepublic void onLoadFinished(@NonNull Loader<Cursor> loader, Cursor data) {// 更新 UI}@Overridepublic void onLoaderReset(@NonNull Loader<Cursor> loader) {// 清理资源}
}

2. 实践案例

2.1 案例一:消息应用的数据同步

在消息应用中,消息数据通常需要在客户端和服务器之间同步。使用 ContentProvider,可以方便地实现数据的本地存储和跨进程访问,同时结合 Loader 和异步任务,确保数据加载和更新的流畅性。

消息 ContentProvider 实现
public class MessageProvider extends ContentProvider {private static final String AUTHORITY = "com.example.provider";private static final String BASE_PATH = "messages";public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/" + BASE_PATH);private static final int MESSAGES = 1;private static final int MESSAGE_ID = 2;private static final UriMatcher uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);static {uriMatcher.addURI(AUTHORITY, BASE_PATH, MESSAGES);uriMatcher.addURI(AUTHORITY, BASE_PATH + "/#", MESSAGE_ID);}private SQLiteDatabase database;@Overridepublic boolean onCreate() {DatabaseHelper dbHelper = new DatabaseHelper(getContext());database = dbHelper.getWritableDatabase();return true;}@Nullable@Overridepublic Cursor query(@NonNull Uri uri, @Nullable String[] projection, @Nullable String selection,@Nullable String[] selectionArgs, @Nullable String sortOrder) {switch (uriMatcher.match(uri)) {case MESSAGES:return database.query(DatabaseHelper.TABLE_MESSAGES, projection, selection, selectionArgs, null, null, sortOrder);case MESSAGE_ID:selection = DatabaseHelper.COLUMN_ID + "=?";selectionArgs = new String[]{String.valueOf(ContentUris.parseId(uri))};return database.query(DatabaseHelper.TABLE_MESSAGES, projection, selection, selectionArgs, null, null, sortOrder);default:throw new IllegalArgumentException("Unknown URI: " + uri);}}@Nullable@Overridepublic Uri insert(@NonNull Uri uri, @Nullable ContentValues values) {long id = database.insert(DatabaseHelper.TABLE_MESSAGES, null, values);getContext().getContentResolver().notifyChange(uri, null);return ContentUris.withAppendedId(CONTENT_URI, id);}@Overridepublic int delete(@NonNull Uri uri, @Nullable String selection, @Nullable String[] selectionArgs) {int rowsDeleted;switch (uriMatcher.match(uri)) {case MESSAGES:rowsDeleted = database.delete(DatabaseHelper.TABLE_MESSAGES, selection, selectionArgs);break;case MESSAGE_ID:selection = DatabaseHelper.COLUMN_ID + "=?";selectionArgs = new String[]{String.valueOf(ContentUris.parseId(uri))};rowsDeleted = database.delete(DatabaseHelper.TABLE_MESSAGES, selection, selectionArgs);break;default:throw new IllegalArgumentException("Unknown URI: " + uri);}getContext().getContentResolver().notifyChange(uri, null);return rowsDeleted;}@Overridepublic int update(@NonNull Uri uri, @Nullable ContentValues values, @Nullable String selection, @Nullable String[] selectionArgs) {int rowsUpdated;switch (uriMatcher.match(uri)) {case MESSAGES:rowsUpdated = database.update(DatabaseHelper.TABLE_MESSAGES, values, selection, selectionArgs);break;case MESSAGE_ID:selection = DatabaseHelper.COLUMN_ID + "=?";selectionArgs = new String[]{String.valueOf(ContentUris.parseId(uri))};rowsUpdated = database.update(DatabaseHelper.TABLE_MESSAGES, values, selection, selectionArgs);break;default:throw new IllegalArgumentException("Unknown URI: " + uri);}getContext().getContentResolver().notifyChange(uri, null);return rowsUpdated;}@Nullable@Overridepublic String getType(@NonNull Uri uri) {switch (uriMatcher.match(uri)) {case MESSAGES:return "vnd.android.cursor.dir/vnd.com.example.provider.messages";case MESSAGE_ID:return "vnd.android.cursor.item/vnd.com.example.provider.message";default:throw new IllegalArgumentException("Unknown URI: " + uri);}}
}
2.2 案例二:音乐播放器的媒体库管理

在音乐播放器应用中,媒体库管理需要高效的数据存储和查询功能。ContentProvider 可以为应用提供统一的数据访问接口,并结合批量操作和事务管理,实现高效的数据管理。

媒体库 ContentProvider 实现
public class MediaProvider extends ContentProvider {private static final String AUTHORITY = "com.example.provider";private static final String BASE_PATH = "media";public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/" + BASE_PATH);private static final int MEDIA = 1;private static final int MEDIA_ID = 2;private static final UriMatcher uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);static {uriMatcher.addURI(AUTHORITY, BASE_PATH, MEDIA);uriMatcher.addURI(AUTHORITY, BASE_PATH + "/#", MEDIA_ID);}private SQLiteDatabase database;@Overridepublic boolean onCreate() {DatabaseHelper dbHelper = new DatabaseHelper(getContext());database = dbHelper.getWritableDatabase();return true;}@Nullable@Overridepublic Cursor query(@NonNull Uri uri, @Nullable String[] projection, @Nullable String selection,@Nullable String[] selectionArgs, @Nullable String sortOrder) {switch (uriMatcher.match(uri)) {case MEDIA:return database.query(DatabaseHelper.TABLE_MEDIA, projection, selection, selectionArgs, null, null, sortOrder);case MEDIA_ID:selection = DatabaseHelper.COLUMN_ID + "=?";selectionArgs = new String[]{String.valueOf(ContentUris.parseId(uri))};return database.query(DatabaseHelper.TABLE_MEDIA, projection, selection, selectionArgs, null, null, sortOrder);default:throw new IllegalArgumentException("Unknown URI: " + uri);}}@Nullable@Overridepublic Uri insert(@NonNull Uri uri, @Nullable ContentValues values) {long id = database.insert(DatabaseHelper.TABLE_MEDIA, null, values);getContext().getContentResolver().notifyChange(uri, null);return ContentUris.withAppendedId(CONTENT_URI, id);}@Overridepublic int delete(@NonNull Uri uri, @Nullable String selection, @Nullable String[] selectionArgs) {int rowsDeleted;switch (uriMatcher.match(uri)) {case MEDIA:rowsDeleted = database.delete(DatabaseHelper.TABLE_MEDIA, selection, selectionArgs);break;case MEDIA_ID:selection = DatabaseHelper.COLUMN_ID + "=?";selectionArgs = new String[]{String.valueOf(ContentUris.parseId(uri))};rowsDeleted = database.delete(DatabaseHelper.TABLE_MEDIA, selection, selectionArgs);break;default:throw new IllegalArgumentException("Unknown URI: " + uri);}getContext().getContentResolver().notifyChange(uri, null);return rowsDeleted;}@Overridepublic int update(@NonNull Uri uri, @Nullable ContentValues values, @Nullable String selection, @Nullable String[] selectionArgs) {int rowsUpdated;switch (uriMatcher.match(uri)) {case MEDIA:rowsUpdated = database.update(DatabaseHelper.TABLE_MEDIA, values, selection, selectionArgs);break;case MEDIA_ID:selection = DatabaseHelper.COLUMN_ID + "=?";selectionArgs = new String[]{String.valueOf(ContentUris.parseId(uri))};rowsUpdated = database.update(DatabaseHelper.TABLE_MEDIA, values, selection, selectionArgs);break;default:throw new IllegalArgumentException("Unknown URI: " + uri);}getContext().getContentResolver().notifyChange(uri, null);return rowsUpdated;}@Nullable@Overridepublic String getType(@NonNull Uri uri) {switch (uriMatcher.match(uri)) {case MEDIA:return "vnd.android.cursor.dir/vnd.com.example.provider.media";case MEDIA_ID:return "vnd.android.cursor.item/vnd.com.example.provider.media";default:throw new IllegalArgumentException("Unknown URI: " + uri);}}
}

3. 总结

ContentProvider 是 Android 中强大的数据共享和管理机制,尤其适用于跨进程数据共享和提供统一的数据访问接口。在实际应用中,通过合理设计和优化,可以充分发挥 ContentProvider 的优势,确保数据操作的高效性和安全性。遵循最佳实践并结合具体场景进行性能优化,可以显著提升应用的用户体验和稳定性。

欢迎点赞|关注|收藏|评论,您的肯定是我创作的动力

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/50588.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux:基础

一、安装 二、 一些组件 2.1 git管理 集中式版本控制系统:版本库是集中存放在中央服务器的,需要时要先从中央服务器取得最新的版本进行修改,修改后再推送给中央服务器。集中式版本控制系统最大的毛病就是必须联网才能工作,网速慢的话影响太大。 分布式版本控制系统:分布…

Linux网络-wget命令

作者介绍&#xff1a;简历上没有一个精通的运维工程师。希望大家多多关注我&#xff0c;我尽量把自己会的都分享给大家&#xff0c;下面的思维导图也是预计更新的内容和当前进度(不定时更新)。 Linux服务器作为一个常用的网络服务器&#xff0c;主要的作用就是向客户端提供网络…

设计模式14-享元模式

设计模式14-享元模式 由来动机定义与结构代码推导特点享元模式的应用总结优点缺点使用享元模式的注意事项 由来动机 在很多应用中&#xff0c;可能会创建大量相似对象&#xff0c;例如在文字处理器中每个字符对象。在这些场景下&#xff0c;如果每个对象都独立存在&#xff0c…

PyCharm 2024.1.4:一站式教程与新特性解析

简介 PyCharm是由JetBrains开发的一款Python集成开发环境&#xff08;IDE&#xff09;&#xff0c;自发布以来&#xff0c;凭借其强大的功能、智能的代码补全、广泛的插件支持和用户友好的界面&#xff0c;成为了Python开发者的首选工具之一。无论是数据科学、Web开发还是其他…

Redis - SpringDataRedis - RedisTemplate

目录 概述 创建项目 引入依赖 配置文件 测试代码 测试结果 数据序列化器 自定义RedisTemplate的序列化方式 测试报错 添加依赖后测试 存入一个 String 类型的数据 测试存入一个对象 优化 -- 手动序列化 测试存入一个Hash 总结&#xff1a; 概述 SpringData 是 S…

在 ArchLinux 上编译运行 axmol 引擎

本文将在 Windows 10 上安装 Arch WSL 中编译 axmol 请确保 WSL2 已正确安装 1. 在微软应用商店安装 ArchLinux 2. 打开 Arch&#xff0c;按照提示输入用户名和密码&#xff0c;尽量简单 3. 配置清华源&#xff0c;速度快的起飞&#xff0c;否则&#xff0c;各种包会安装失败…

光伏电站气象站:现代光伏系统的重要组成部分

光伏电站气象站&#xff0c;作为现代光伏系统的重要组成部分&#xff0c;集成了气象学、电子信息技术、数据处理与分析等多学科技术于一体&#xff0c;能够实时监测并记录包括温度、湿度、风速、风向、太阳辐射强度、降雨量在内的多种气象参数。这些数据不仅是评估光伏板发电效…

GLSL教程 第8章:几何着色器

目录 8.1 几何着色器的介绍 几何着色器的主要功能&#xff1a; 几何着色器的工作流程&#xff1a; 8.2 实现基本的几何变换 示例&#xff1a;将三角形扩展成多个三角形 8.3 几何着色器的高级应用 1. 粒子系统 2. 光晕效果 3. 线框模型 小结 几何着色器是图形管线中的一…

应用层自定义协议以及序列化和反序列化

文章目录 应用层自定义协议以及序列化和反序列化1、应用层自定义协议1.1、应用层1.2、协议 2、序列化和反序列化3、TCP 为什么支持全双工4、jsoncpp基础4.1、序列化4.2、反序列化 5、实现网络版计算器6、手写序列化和反序列化 应用层自定义协议以及序列化和反序列化 1、应用层…

爬取贴吧的标题和链接

免责声明 感谢您学习本爬虫学习Demo。在使用本Demo之前&#xff0c;请仔细阅读以下免责声明&#xff1a; 学习和研究目的&#xff1a;本爬虫Demo仅供学习和研究使用。用户不得将其用于任何商业用途或其他未经授权的行为。合法性&#xff1a;用户在使用本Demo时&#xff0c;应确…

智能算法驱动的爬虫平台:解锁网络数据的无限潜力

摘要 在信息爆炸的时代&#xff0c;网络数据如同深海宝藏&#xff0c;等待着有识之士发掘其无尽价值。本文将探索智能算法驱动的爬虫平台如何成为解锁这一宝库的关键&#xff0c;不仅剖析其技术优势&#xff0c;还通过实例展示它如何助力企业与开发者高效、稳定地采集数据&…

C语言 ——— 数组指针的定义 数组指针的使用

目录 前言 数组指针的定义 数组指针的使用 前言 之前有编写过关于 指针数组 的相关知识 C语言 ——— 指针数组 & 指针数组模拟二维整型数组-CSDN博客 指针数组 顾名思义就是 存放指针的数组 那什么是数组指针呢&#xff1f; 数组指针的定义 何为数组指针&#xf…

【QT】UDP

目录 核心API 示例&#xff1a;回显服务器 服务器端编写&#xff1a; 第一步&#xff1a;创建出socket对象 第二步&#xff1a; 连接信号槽 第三步&#xff1a;绑定端口号 第四步&#xff1a;编写信号槽所绑定方法 第五步&#xff1a;编写第四步中处理请求的方法 客户端…

JAVA开发工具IDEA如何连接操作数据库

一、下载驱动 下载地址&#xff1a;【免费】mysql-connector-j-8.2.0.jar资源-CSDN文库 二、导入驱动 鼠标右击下载到IDEA中的jar包&#xff0c;选择Add as Library选项 如图就导入成功 三、加载驱动 Class.forName("com.mysql.cj.jdbc.Driver"); 四、驱动管理…

【C++】——红黑树(手撕红黑树,彻底弄懂红黑树)

目录 前言 一 红黑树简介 二 为什么需要红黑树 三 红黑树的特性 四 红黑树的操作 4.1 变色操作 4.2 旋转操作 4.3 插入操作 4.4 红黑树插入代码实现 4.5 红黑树的删除 五 红黑树迭代器实现 总结 前言 我们之前都学过ALV树&#xff0c;AVL树的本质就是一颗平…

计算机实验室排课查询小程序的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;学生管理&#xff0c;教师管理&#xff0c;实验室信息管理&#xff0c;实验室预约管理&#xff0c;取消预约管理&#xff0c;实验课程管理&#xff0c;实验报告管理&#xff0c;报修信息管理&#xff0…

Linux的yum源安装MySQL5.7

linux的yum源安装MySQL5.7 一、MySQL 1、简介 MySQL 是一种流行的关系型数据库管理系统&#xff08;RDBMS&#xff09;&#xff0c;由瑞典公司 MySQL AB 开发&#xff0c;后来被 Oracle Corporation 收购。它是一个开源软件&#xff0c;提供了高效、稳定和可靠的数据管理解决…

Spring AI (三) 提示词对象Prompt

3.提示词对象Prompt 3.1.Prompt Prompt类的作用是创建结构化提示词, 实现了ModelRequest<List<Message>>接口 Prompt(String contents)&#xff1a;创建一个包含指定内容的Prompt对象。 Prompt(String contents, ChatOptions modelOptions)&#xff1a;创建一个…

某数据泄露防护(DLP)系统NoticeAjax接口SQL注入漏洞复现 [附POC]

文章目录 某数据泄露防护(DLP)系统NoticeAjax接口SQL注入漏洞复现 [附POC]0x01 前言0x02 漏洞描述0x03 影响版本0x04 漏洞环境0x05 漏洞复现1.访问漏洞环境2.构造POC3.复现某数据泄露防护(DLP)系统NoticeAjax接口SQL注入漏洞复现 [附POC] 0x01 前言 免责声明:请勿利用文章内…

DolphinDB Web 端权限管理:可视化操作指南

在现代数据库管理中&#xff0c;高效和直观的权限管理对于用户的数据安全是至关重要的。过去 DolphinDB 用户需要依赖系统脚本来管理用户和权限&#xff0c;这对于缺乏技术背景的管理员来说既复杂又容易出错。 为了提升用户体验和操作效率&#xff0c;DolphinDB 目前在 Web 上…