ContentProvider数据库共享之概述
前言:项目终于结束了,终于有自己的时间了,感觉几个月没什么进展了,毕业的时间越长,反而觉得学到的知识越少。努力吧。 1、《ContentProvider数据库共享之——概述》 2、《ContentProvider数据库共享之——实例讲解》 3、《ContentProvider数据库共享之—
前言:项目终于结束了,终于有自己的时间了,感觉几个月没什么进展了,毕业的时间越长,反而觉得学到的知识越少。努力吧。
1、《ContentProvider数据库共享之——概述》
2、《ContentProvider数据库共享之——实例讲解》
3、《ContentProvider数据库共享之——MIME类型与getType()》
4、《ContentProvider数据库共享之——读写权限与数据监听》
这篇主要给大家全局性的讲讲应用间的数据库共享。我们都知道每个应用的数据库都只允许创建它的应用自己来读写,那其它应用要访问这个数据库要怎么办呢;这就是数据库共享。
举个例子:如果有写了两个姊妹应用,比如,阿里旺旺和淘宝;如果我想在用淘宝的时候就直接想看到旺旺里跟哪个卖家的聊天记录怎么办(不考虑网络获取,只考虑本地共享),这时候我就要去旺旺的数据库里去找了吧。这就是两应用间的数据库共享问题。而ContentProvider就是解决这个问题的。下面就从ContentProvider设计者的角度来讲讲这两个不同应用间数据库共享问题。
一、两应用间如何通信
首先,我们最先想到的方法应该是如下面这张图这样的,自身应用写好各个数据库接口,供其它应用调用
那么,问题来了,首先,其它应用如何调起这个应用的数据库接口;要知道你要调这个接口的应用很可能根本就没有运行。
咦,好像有一种方法就能调起没有运行的应用——隐式Intent!!!
隐式Intent的调用方法如下,如果这个Activity要供其它应用调用,那么这个Activity在这个应用AndroidManifest.xml中声明方式为:
<activity android:name=".SecondActivity" android:label="@string/title_activity_second"> <intent-filter> <action android:name="harvic.test.qijian"></action> <category android:name="android.intent.category.DEFAULT"></category> </intent-filter> </activity>
Intent intent = new Intent("harvic.test.qijian"); startActivity(intent);
1、首先,我们将filter信息写在了AndroidManifest.xml中;
2、当有一个隐式Intent要匹配时,系统(注意是系统!)会搜索整个手机上所有APP的Acitivity进行匹配,如果有匹配的并且有使用权限的,就起起来;大家想想当你在应用中点击网址链接的时候,是不是会转到浏览器中,这就是用的隐式Intent匹配。如果你手机上有多个应用可以匹配,那么就会以列表的形式列出来供你选择开哪一个。具体有关隐式Intent的内容,可以参看《Intent详解一》和《Intent详解二》
根据隐式Intent,我们是不是有受到启发,那我们也可额外再开出来一个东东,仿照隐式Intent的方式来进行全局匹配,如果匹配成功就执行操作!!!对,这就是那帮谷歌老头的设计方法,他们设计的东东就是ContentProvider的URI!!二、ContentProvider与对应的URI
ContentProvider的URI就是下面这个形式的:
主要分三个部分:scheme, authority and path。scheme表示上图中的content://,authority表示B部分,path表示C和D部分。
A部分:表示是一个Android内容URI,说明由ContentProvider控制数据,该部分是固定形式,不可更改的。
B部分:是URI的授权部分,是唯一标识符,用来定位ContentProvider。格式一般是自定义ContentProvider类的完全限定名称,注册时需要用到,如:com.example.transportationprovider
C部分和D部分:是每个ContentProvider内部的路径部分,C和D部分称为路径片段,C部分指向一个对象集合,一般用表的名字,如:/trains表示一个笔记集合;D部分指向特定的记录,如:/trains/122表示id为122的单条记录,如果没有指定D部分,则返回全部记录。
这个URI就是用来进行全局匹配的,那AndroidManifest.xml里又要怎么写呢?
再回想下隐式Intent,在隐式Intent中,Intent-fliter是Activity的一部分,专门过滤隐式Intent的匹配请求的,如果Intent-fliter匹配后,就启动对应的Activity;
那我们也可以设计一个东东,把这个URI作为它的一部分,当匹配成功以后,就进这个东东里操作数据库。这个东东就是ContentProvider;
ContentProvider在AndroidManifest.xml中的声明方式为:(与上面的URI对应)
<provider android:name=".NoteContentProvider" android:authorities="com.example.transportationprovider" android:exported="true"></provider>
这里的android:authorities必须与上面URI中的B部分一样,因为这个就是用来全局匹配的authority!!!!只有URI的authority与provider的android:authorities匹配上了,才会进入后面的操作
到这里,我也要祭出大招了:ContentURI的全局流程图
上面,我们说到了第一步和第二步,当匹配成功ContentProvider以后,就开始进入我们指定的ContentProvider进行处理;大家估计也注意到了,我们的ContentURI的完整部分为:content://com.example.transportionprovider/trains/122
到第二步,我们已经匹配到了content://com.example.transportionprovider,那后面的/trains/122的匹配工作就只有交由ContentProvider来处理了。
在ContentProvder中的匹配是利用UriMatcher来完成的!
三、UriMatcher
UriMatcher的匹配工作的第一步就是先将所需要的匹配的URI使用addURI()添加到UriMatcher中,
public void addURI(String authority, String path, int code)
其中第一个参数authority:就是URI对应的authority
path:就是我们在URI中 authority后的那一串
code:表示匹配成功以后的返回值;
下面以我们的URI=content://com.example.transportionprovider/trains/122来演示一下
public static final String AUTHORITY = "com.alexzhou.provider.NoteProvider"; private static final UriMatcher sUriMatcher; static { sUriMatcher = new UriMatcher(UriMatcher.NO_MATCH); sUriMatcher.addURI(AUTHORITY, "trains", 1); sUriMatcher.addURI(AUTHORITY, "trains/#", 2); }
上段代码中:
sUriMatcher.addURI(AUTHORITY, "trains", 1);
表示匹配content://com.example.transportionprovider/trains,如果匹配成功返回1
sUriMatcher.addURI(AUTHORITY, "trains/#", 2);
其中
#表示匹配任意数据ID
*表示匹配任意文本
所以这句的意思就是匹配content://com.example.transportionprovider/trains/任意数字ID 比如我们的content://com.example.transportionprovider/trains/122
在匹配以后就是响应请求啦。至于如何响应请求就在下节在讲了,这节就主要把流程给大家讲清楚及涉及到有知识,到这,整体流程就讲完了,但还有个地方还没说,就是第三方应用如何根据URI来指定操作的,是哪个函数来操作URI的呢?
它就是ContentResolver;
下面简单先说一下ContentResolver的函数:
插入(insert):
String CONTENT_URI = content://com.example.transportionprovider/trains/122; ContentResolver cr =getContentResolver(); ContentValues values = new ContentValues(); values.put("title", "hello");//数据库的键值对 values.put("content", "my name is harvic"); Uri uri = cr.insert(CONTENT_URI, values);
在第三方中,我们要向指定应用的数据库中插入一条记录,其中title字段的值为hello,content字段的值为my name is harvic。就是用上面的这一段代码,有没有很简单
这段代码一调用,那可就吊了,系统会搜索手机上所有APP的AndroidManifest.xml,看哪个provider的authority匹配,在匹配之后,转到对应的类中,再让UriMatcher匹配后面的PATH字段,都完全匹配之后,就执行ContentProvider中的insert方法!!!!这就是是整个流程。
当然ContentResolver除了insert方法还有query()、update()、delete()方法,可以执行第三方数据库的任何操作。
好了,到这整个流程就讲完,想必大家对ContentProvider涉及到的几部分也都清楚了吧。下篇给大家用实例讲讲,他们是如何工作的。
http://blog.csdn.net/harvic880925/article/details/44521461 谢谢!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

苹果公司最新发布的iOS18、iPadOS18以及macOSSequoia系统为Photos应用增添了一项重要功能,旨在帮助用户轻松恢复因各种原因丢失或损坏的照片和视频。这项新功能在Photos应用的"工具"部分引入了一个名为"已恢复"的相册,当用户设备中存在未纳入其照片库的图片或视频时,该相册将自动显示。"已恢复"相册的出现为因数据库损坏、相机应用未正确保存至照片库或第三方应用管理照片库时照片和视频丢失提供了解决方案。用户只需简单几步

Hibernate多态映射可映射继承类到数据库,提供以下映射类型:joined-subclass:为子类创建单独表,包含父类所有列。table-per-class:为子类创建单独表,仅包含子类特有列。union-subclass:类似joined-subclass,但父类表联合所有子类列。

如何在PHP中使用MySQLi建立数据库连接:包含MySQLi扩展(require_once)创建连接函数(functionconnect_to_db)调用连接函数($conn=connect_to_db())执行查询($result=$conn->query())关闭连接($conn->close())

PHP中处理数据库连接报错,可以使用以下步骤:使用mysqli_connect_errno()获取错误代码。使用mysqli_connect_error()获取错误消息。通过捕获并记录这些错误信息,可以轻松识别并解决数据库连接问题,确保应用程序的顺畅运行。

HTML无法直接读取数据库,但可以通过JavaScript和AJAX实现。其步骤包括建立数据库连接、发送查询、处理响应和更新页面。本文提供了利用JavaScript、AJAX和PHP来从MySQL数据库读取数据的实战示例,展示了如何在HTML页面中动态显示查询结果。该示例使用XMLHttpRequest建立数据库连接,发送查询并处理响应,从而将数据填充到页面元素中,实现了HTML读取数据库的功能。

通过Go标准库database/sql包,可以连接到MySQL、PostgreSQL或SQLite等远程数据库:创建包含数据库连接信息的连接字符串。使用sql.Open()函数打开数据库连接。执行SQL查询和插入操作等数据库操作。使用defer关闭数据库连接以释放资源。

在Golang中使用数据库回调函数可以实现:在指定数据库操作完成后执行自定义代码。通过单独的函数添加自定义行为,无需编写额外代码。回调函数可用于插入、更新、删除和查询操作。必须使用sql.Exec、sql.QueryRow或sql.Query函数才能使用回调函数。

可以通过使用gjson库或json.Unmarshal函数将JSON数据保存到MySQL数据库中。gjson库提供了方便的方法来解析JSON字段,而json.Unmarshal函数需要一个目标类型指针来解组JSON数据。这两种方法都需要准备SQL语句和执行插入操作来将数据持久化到数据库中。
