Android 这 13 道 ContentProvider 面试题,你都会了吗?
前言
- 作为
Android的四大组件之一,ContentProvider可以说是无处不在了。 - 但是对于我而言,开发过程中看似
ContentProvider用得很娴熟,却一直没能形成一个完整的体系。 - 也许大家也有着和我类似的烦恼,于是我特地花了几天的时间,总结了我所知道的知识点,以及面试中可能遇到的问题。将本文分享给大家,希望能帮助大家重新梳理下我们的这个老朋友
ContentProvider。
最后,希望大家阅读愉快!
文章目录
ContentProvider应用程序间非常通用的共享数据的一种方式,也是Android官方推荐的方式。Android中许多系统应用都使用该方式实现数据共享,比如通讯录、短信等。
方便大家学习,我在 GitHub 上建立个 仓库
仓库内容与博客同步更新。由于我在
稀土掘金简书CSDN博客园等站点,都有新内容发布。所以大家可以直接关注该仓库,以免错过精彩内容!
1.1 Android 为什么要设计 ContentProvider 这个组件?
- 很多做
Android开发的人都不怎么使用它,觉得直接读取数据库会更简单方便。 - 那么
Android搞一个内容提供者在数据和应用之间,只是为了装高大上,故弄玄虚?其设计用意在于:
- 封装。对数据进行封装,提供统一的接口,使用者完全不必关心这些数据是在
DB,XML、Preferences或者网络请求来的。当项目需求要改变数据来源时,使用我们的地方完全不需要修改。 - 提供一种跨进程数据共享的方式。
- 应用程序间的数据共享还有另外的一个重要话题,就是数据更新通知机制了。因为数据是在多个应用程序中共享的,当其中一个应用程序改变了这些共享数据的时候,它有责任通知其它应用程序,让它们知道共享数据被修改了,这样它们就可以作相应的处理。
1.2 如何访问自定义 ContentProvider
ContentResolver接口的notifyChange函数来通知那些注册了监控特定 URI的ContentObserver 对象,使得它们可以相应地执行一些处理。- ContentObserver 可以通过 registerContentObserver 进行注册。
- 通过
ContentProvider的Uri访问开放的数据。
ContenResolver对象通过Context提供的方法getContenResolver()来获得。ContenResolver提供了以下方法来操作:insertdeleteupdatequery这些方法分别会调用ContenProvider中与之对应的方法并得到返回的结果。
1.3 通过 ContentResolver 获取 ContentProvider 内容的基本步骤
- 得到
ContentResolver类对象:ContentResolver cr = getContentResolver ( )。 - 定义要查询的字段
String数组。 - 使用
cr.query(); 返回一个Cursor对象。 - 使用
while循环得到Cursor里面的内容。
1.4 ContentProvider 是如何实现数据共享的:
- 在
Android中如果想将自己应用的数据 ( 一般多为数据库中的数据 ) 提供给第三发应用, 那么我们只能通过ContentProvider来实现了。ContentProvider是应用程序之间共享数据的接口。 - 使用的时候首先自定义 一个类继承
ContentProvider, 然后覆写query、insert、update、delete等 方法。 - 因为其是四大组件之一因此必须在
AndroidManifest文件中进行注册。 - 把自己的数据通过
uri的形式共享出去android系统下 不同程序 数据默认是不能共享访问 需要去实现一个类去继承ContentProvider。
public class PersonContentProvider extends ContentProvider{
public boolean onCreate(){ }
query(Url, String[], String, String[], String);
insert(Uri,ContentValues);
update(Uri,ContentValues,String[]);
delete(Uri,String,String[]);
}
1.5 为什么要用 ContentProvider ?它和 sql 的实现上有什么差别?
ContentProvider屏蔽了数据存储的细节 , 内部实现对用户完全透明 , 用户只需要关心操作数据的uri就可以了,ContentProvider可以实现不同app之间 共享。Sql也有增删改查的方法, 但是sql只能查询本应用下的数据库。- 而
ContentProvider还可以去增删改查本地文件.xml文件的读取等。
1.6 Uri 介绍
- 为系统的每一个资源给其一个名字,比方说通话记录。
- 每一个
ContentProvider都拥有一个公共的URI,这个URI用于表示这个ContentProvider所提供的数据。 Android所提供的ContentProvider都存放在android.provider包中。
- 将其分为
A,B,C,D4个部分: A:标准前缀,用来说明一个Content Provider控制这些数据,无法改变的;"content://";B:URI的标识,用于唯一标识这个ContentProvider,外部调用者可以根据这个标识来找到它。它定义了是哪个ContentProvider提供这些数据。对于第三方应用程序,为了保证URI标识的唯一性,它必须是一个完整的、小写的类名。这个标识在元素的authorities属性中说明:一般是定义该ContentProvider的包类的名称;C:路径(path),通俗的讲就是你要操作的数据库中表的名字,或者你也可以自己定义,记得在使用的时候保持一致就可以了;"content://com.bing.provider.myprovider/tablename"。D:如果URI中包含表示需要获取的记录的ID;则就返回该id对应的数据,如果没有ID,就表示返回全部;"content://com.bing.provider.myprovider/tablename/#"#表示数据id。
1.7 如何访问 asserts 资源目录下的数据库?
- 把数据库
db复制到/data/data/packagename/databases/目录下, 然后直接就能访问了。
1.8 多个进程同时调用一个 ContentProvider 的 query 获取数据,ContentPrvoider 是如何反应的呢?
- 一个
ContentProvider可以接受来自另外一个进程的数据请求。 - 尽管
ContentResolver与ContentProvider类隐藏了实现细节,但是ContentProvider所提供的query(),insert(),delete(),update()都是在ContentProvider进程的线程池中被调用执行的,而不是进程的主线程中。 - 这个线程池是有
Binder创建和维护的,其实使用的就是每个应用进程中的Binder线程池。
1.9 Android 设计 ContentProvider 的目的是什么呢?
- 隐藏数据的实现方式,对外提供统一的数据访问接口;
- 更好的数据访问权限管理。
ContentProvider可以对开发的数据进行权限设置,不同的URI可以对应不同的权限,只有符合权限要求的组件才能访问到ContentProvider的具体操作。 ContentProvider封装了跨进程共享的逻辑,我们只需要Uri即可访问数据。由系统来管理ContentProvider的创建、生命周期及访问的线程分配,简化我们在应用间共享数据( 进程间通信 )的方式。我们只管通过ContentResolver访问ContentProvider所提示的数据接口,而不需要担心它所在进程是启动还是未启动。
1.10 运行在主线程的 ContentProvider 为什么不会影响主线程的 UI 操作?
ContentProvider的onCreate()是运行在UI线程的- 而
query(),insert(),delete(),update()是运行在线程池中的工作线程的 - 所以调用这向个方法并不会阻塞
ContentProvider所在进程的主线程,但可能会阻塞调用者所在的进程的UI线程! - 所以,调用
ContentProvider的操作仍然要放在子线程中去做。 - 虽然直接的
CRUD的操作是在工作线程的,但系统会让你的调用线程等待这个异步的操作完成,你才可以继续线程之前的工作。
1.11 外提供数据共享,那么如何限制对方的使用呢?
android:exported属性非常重要。这个属性用于指示该服务是否能够被其他应用程序组件调用或跟它交互。如果设置为
true,则能够被调用或交互,否则不能。设置为
false时,只有同一个应用程序的组件或带有相同用户ID的应用程序才能启动或绑定该服务。对于需要开放的组件应设置合理的权限,如果只需要对同一个签名的其它应用开放
ContentProvider,则可以设置signature级别的权限。大家可以参考一下系统自带应用的代码,自定义了
signature级别的permission:
<permission android:name="com.android.gallery3d.filtershow.permission.READ"
android:protectionLevel="signature" />
<permission android:name="com.android.gallery3d.filtershow.permission.WRITE"
android:protectionLevel="signature" />
<provider
android:name="com.android.gallery3d.filtershow.provider.SharedImageProvider"
android:authorities="com.android.gallery3d.filtershow.provider.SharedImageProvider"
android:grantUriPermissions="true"
android:readPermission="com.android.gallery3d.filtershow.permission.READ"
android:writePermission="com.android.gallery3d.filtershow.permission.WRITE" />
1.11.1 如果我们只需要开放部份的 URI 给其他的应用访问呢?
- 可以参考
Provider的URI权限设置,只允许访问部份URI,可以参考原生ContactsProvider2的相关代码( 注意path-permission这个选项 ):
<provider android:name="ContactsProvider2"
android:authorities="contacts;com.android.contacts"
android:label="@string/provider_label"
android:multiprocess="false"
android:exported="true"
android:grantUriPermissions="true"
android:readPermission="android.permission.READ_CONTACTS"
android:writePermission="android.permission.WRITE_CONTACTS">
<path-permission
android:pathPrefix="/search_suggest_query"
android:readPermission="android.permission.GLOBAL_SEARCH" />
<path-permission
android:pathPrefix="/search_suggest_shortcut"
android:readPermission="android.permission.GLOBAL_SEARCH" />
<path-permission
android:pathPattern="/contacts/.*/photo"
android:readPermission="android.permission.GLOBAL_SEARCH" />
<grant-uri-permission android:pathPattern=".*" />
</provider>
1.12 ContentProvider 接口方法运行在哪个线程中呢?
ContentProvider可以在AndroidManifest.xml中配置一个叫做android:multiprocess的属性,默认值是 false ,表示 ContentProvider 是单例的- 无论哪个客户端应用的访问都将是同一个
ContentProvider对象,如果设为true,系统会为每一个访问该ContentProvider的进程创建一个实例。
1.12.1 这点还是比较好理解的,那如果我要问每个 ContentProvider 的操作是在哪个线程中运行的呢?( 其实我们关心的是 UI 线程和工作线程 )
比如我们在UI线程调用getContentResolver().query查询数据,而当数据量很大时(或者需要进行较长时间的计算)会不会阻塞UI线程呢?
要分两种情况
ContentProvider和调用者在同一个进程,ContentProvider的方法(query/insert/update/delete等 )和调用者在同一线程中;ContentProvider和调用者在不同的进程,ContentProvider的方法会运行在它自身所在进程的一个 Binder 线程中。
但是,注意这两种方式在ContentProvider的方法没有执行完成前都会blocked调用者。所以你应该知道这个上面这个问题的答案了吧。- 也可以看看
CursorLoader这个类的源码,看Google自己是怎么使用getContentResolver().query的。
1.13 ContentProvider 是如何在不同应用程序之间传输数据的?
- 一个应用进程有
16个Binder线程去和远程服务进行交互,而每个线程可占用的缓存空间是128KB这样,超过会报异常。 ContentResolver虽然是通过Binder进程间通信机制打通了应用程序之间共享数据的通道,但ContentProvider组件在不同应用程序之间传输数据是基于匿名共享内存机制来实现的。
总结
- 在这篇文章中,我对我所知道的
ContentProvider知识总进行了详细的总结,希望大家通过本次阅读都能有所收获。 重点:学Android有一段时间了,我打算好好的梳理一下所学知识,到现在为止,我才总结完Activity、Service、BroadcastRecevier等,有关 事件分发、滑动冲突、新能优化等重要模块,我后面也将详尽的总结,欢迎大家关注 _yuanhao 的 博客园 ,方便及时接收更新- 如果有可以补充的知识点,欢迎大家在评论区指出。
码字不易,你的点赞是我总结的最大动力!
由于我在「稀土掘金」「简书」「
CSDN」「博客园」等站点,都有新内容发布。所以大家可以直接关注我的GitHub仓库,以免错过精彩内容!一万多字长文,加上精美思维导图,记得点赞哦,欢迎关注 _yuanhao 的 博客园 ,我们下篇文章见!
Android 这 13 道 ContentProvider 面试题,你都会了吗?的更多相关文章
- 这10道javascript笔试题你都会么
1.考察this var length = 10; function fn() { console.log(this.length); } var obj = { length: 5, metho ...
- 你应该知道的25道Javascript面试题
题目来自 25 Essential JavaScript Interview Questions.闲来无事,正好切一下. 一 What is a potential pitfall with usin ...
- [ZZ]知名互联网公司Python的16道经典面试题及答案
知名互联网公司Python的16道经典面试题及答案 https://mp.weixin.qq.com/s/To0kYQk6ivYL1Lr8aGlEUw 知名互联网公司Python的16道经典面试题及答 ...
- Android 四大组件之" ContentProvider "
前言 ContentProvider作为Android的四大组件之一,是属于需要掌握的基础知识,可能在我们的应用中,对于Activity和Service这两个组件用的很常见,了解的也很多,但是对Con ...
- MySql_34道经典Sql试题
MySql_34道经典Sql试题 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/xiaouncle/article/details/799390 ...
- 【笔试题】精选30道Java笔试题解答
转自于:精选30道Java笔试题解答 精选30道Java笔试题解答 1. 下面哪些是Thread类的方法() A. start() B. run() C. exit() D. getPriority( ...
- 精选30道Java笔试题附答案分析
精选30道Java笔试题解答 都是一些非常非常基础的题,是我最近参加各大IT公司笔试后靠记忆记下来的,经过整理献给与我一样参加各大IT校园招聘的同学们,纯考Java基础功底,老手们就不用进来了,免得笑 ...
- 49道Spring面试题和答案
49道Spring面试题和答案 Spring 概述 1. 什么是spring? Spring 是个Java企业级应用的开源开发框架.Spring主要用来开发Java应用,但是有些扩展是针对构建J2EE ...
- 史上最全 69 道 Spring 面试题和答案
史上最全 69 道 Spring 面试题和答案 目录Spring 概述依赖注入Spring beansSpring注解Spring数据访问Spring面向切面编程(AOP)Spring MVC Spr ...
随机推荐
- 多线程基础(主要内容转载于https://segmentfault.com/a/1190000014428190)
进程作为资源分配的基本单位 线程作为资源调度的基本单位,是程序的执行单元,执行路径(单线程:一条执行路径,多线程:多条执行路径).是程序使用CPU的最基本单位. 线程有3个基本状态: 执行.就绪.阻塞 ...
- [WP8.1]RSA 使用BouncyCastle 公钥解密
写应用的时候遇到个服务器返回私钥加密过的数据 ,然后要在客户端用公钥解密的需求 ,一直没找到方法,应用搁置了一个学期,多方搜索,结论就是.net没有实现公钥解密的方法,要自己实现,于是硬着头皮开始看 ...
- WebSocket协议与抓包
WebSocket协议 WebSocket并不是全新的协议,而是利用了HTTP协议来建立连接,它的目的是在浏览器和服务器之间建立一个不受限的双向通信的通道,比如说,服务器可以在任意时刻发送消息给浏览器 ...
- spring boot项目下application.properties中使用logging.path和logging.file时的细节
logging.path仅仅用于指定日志输出的目录,且不能指定输出的文件名,且默认名为spring.log 若指定的是相对目录,则会生成在当前总项目的目录下 idea中新建sprnig boot项目 ...
- Spring Boot (十四): 响应式编程以及 Spring Boot Webflux 快速入门
1. 什么是响应式编程 在计算机中,响应式编程或反应式编程(英语:Reactive programming)是一种面向数据流和变化传播的编程范式.这意味着可以在编程语言中很方便地表达静态或动态的数据流 ...
- 02-15 Logistic回归(鸢尾花分类)
目录 Logistic回归(鸢尾花分类) 一.导入模块 二.获取数据 三.构建决策边界 四.训练模型 4.1 C参数与权重系数的关系 五.可视化 更新.更全的<机器学习>的更新网站,更有p ...
- Mapper
public static T MapTo<T>(this object obj) { if (obj == null) return default(T); Mapper.CreateM ...
- iptables详解之filter
iptables详解之filter iptables令很多小伙伴脑阔疼,下面我们来说说如何使用iptables. 一.iptables格式 1.1.iptables 帮助 通过iptables --h ...
- flask 微电影网站
flask简介 轻量级web应用框架 WSGI工具箱才用Werkzeug 模版引擎则使用Jinja2 Flask使用BSD授权 1.virtualenv的使用 (1)创建虚拟环境:virtualenv ...
- django搭建BBS-登入&验证码的生成
django搭建BBS-登入&验证码的生成 基于注册完成后 文件结构 app 接口 migrations __inint__.py admin.py 管理员页面注册表单用 apps.py bb ...