什么是内容提供器?


跨程序共享数据之内容提供器,这是个什么功能?看到这个名称的时候最能给我们提供信息的应该是“跨程序”这个词了,是的重点就是这个词,这个内容提供器的作用主要是用于在不同的引用程序之间实现数据共享的功能,它提供了一完整的机制,允许一个程序访问另一个程序中的数据,同时还能保证被访问的数据的安全性,在目前使用内容提供器是Android实现跨程序共享数据的标准方式。不同于文件存储和SharePreferences存储中的两种全局可读性操作模式,内容提供器可以选择只对那一部分数据进行共享,从而保证我们程序中的隐私数据不糊有泄漏的风险。

不过在理解这个内容提供器之前,我们需要理解Android的运行时权限,这个就不需要我们解释什么是运行时权限了,因为我们在之前其实已经使用过,比如“相机权限”,“照片权限”,“位置权限”等等!

运行时权限


Android 将所有的权限大致的分为两类,一类是普通权限,另一类是危险权限,我们在下面将危险的权限整理了出来,供以后我们参考使用:

举个小栗子


下面是针对打电话我们写的一个小Demo,其实逻辑很简单,说的直接点就是一句话“有权限直接打电话,没有权限就请求完了再打”。下面是点击事件我们做的操作

        // 利用checkSelfPermission这个函数检查有没有运行时权限
// 有权限就直接调用下面的call()方法,没有就请求权限
// checkSelfPermission(MainActivity.this, android.Manifest.permission.CALL_PHONE) 是否等于PackageManager.PERMISSION_GRANTED
// PERMISSION_GRANTED 同意权限
if (ContextCompat.checkSelfPermission(MainActivity.this, android.Manifest.permission.CALL_PHONE) != PackageManager.PERMISSION_GRANTED){ // ActivityCompat的requestPermissions请求权限
// 第二个参数是一个String数组,我们需要把申请的权限名称放到数组中即可
// 第三个参数是请求码,这个请求码我们在下面权限回调的时候可以用来做判断,判断是那个权限再做相应的操作
ActivityCompat.requestPermissions(MainActivity.this,new String[]{ android.Manifest.permission.CALL_PHONE},1);
}else {
call();
}

接着就是我们打电话的call()方法的操作,以及最后权限请求回来之后的回调方法:

    // 防止有异常发生,写在这里
public void call(){
try {
Intent intent = new Intent(Intent.ACTION_CALL);
intent.setData(Uri.parse("tel:10086"));
startActivity(intent);
}catch (SecurityException e){ e.printStackTrace();
}
} //所有的权限回调都是在这里,我们根据requestCode来判断是哪个权限
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
switch (requestCode){
case 1:
if (grantResults.length>0 && grantResults[0] == PackageManager.PERMISSION_GRANTED){
call();
}else {
Toast.makeText(this,"打电话需要权限才能使用",Toast.LENGTH_LONG).show();
}
}
}

访问其他程序中的数据需要-ContentResolver


对于每一个应用程序来说,如果你想要访问内容提供器当中共享的数据,就一定要借助 ContentResolver 类,可以通过Context类当中的getContentResolver()方法来获取该类的实例, ContentResolver 类当中也提供了一系列的方法用于对数据进行CRUD的操作, insert() 添加  update() 更新  delete()  删除   query() 查询

不同于SQLiteDatabase,ContentResolver的CRUD的操作是不接收表名参数的,而是使用一个Uri参数表示。这个参数被称为内容URI,内容URI给内容提供器中的数据表建立了唯一的标识符,它主要是由两部分组成,一部分是 authority,它是用于对不同的应用程序做区分,一般为了避免冲突,都会采取程序包的方式来进行命名, 另一部分是path,path则是相对于同一应用程序中的表走区分的,通常都是添加在authority的后面。当然只有这两部分还是不够的,我们需要在前面加上协议声明,因此标准的形式我们举个例子:

content://com.example.app.provider/table1      当中content://是我们头部的协议      com.example.app.provider是authority   /table1就是path

上面我们得到一个URI字符串之后,我们还需要将它解析成Uri对象才能使用,解析的方法也很简单如下:

Uri uri = Uri.parse("content://com.example.app.provider/table1")

有了这个Uri这个对象之后,我们就可以利用它来查询数据了,代码如下:

Cursor cursor = getContentResolver().query(

uri,                                指定查询某一个应用下面的某张表

projection,                     指定查询的列名

selection,                       指定where的约束条件

selectionArgs,                为where中的占位符提供具体的值

sortOrder);                    指定查询结果的排序方式

查询完成之后返回的仍然是一个Cursor对象,这时候我们就可以将数据从Cursor对象中逐个读取出来了,

读取的思路仍然是通过移动游标的位置来遍历Cursor的所有行没然后再取出每一个行中国的数据没代码如下:

if(cursor != null){

while(cursor.moveNext()){

String column1 = cursor.getString(cursor.getColumnIndex("column1"));

Int column1 = cursor.getInt(cursor.getColumnIndex("column1"));

}

}

掌握了比较复杂的查询之后,剩下的增加,删除,修改就比较简单了,我们也就不在说了!

我们读取一下联系人


我们这里写一个小demo,来读取一下联系人的信息,然后把它展示在一个ListView当中,具体的代码如下:

public class ContactsActivity extends AppCompatActivity {

    //适配器和一个数组,用来存储联系人信息
ArrayAdapter<String> adapter;
List<String> list = new ArrayList<>(); @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_contacts); ListView contactsList = (ListView) findViewById(R.id.contactsListView);
adapter = new ArrayAdapter<String>(ContactsActivity.this, android.R.layout.simple_list_item_1, list);
contactsList.setAdapter(adapter); // 先检查有没有获取通讯录的权限,要是没有就先请求权限
if (ContextCompat.checkSelfPermission(this, android.Manifest.permission.READ_CONTACTS) != PackageManager.PERMISSION_GRANTED){
ActivityCompat.requestPermissions(this,new String[]{android.Manifest.permission.READ_CONTACTS},2);
}else {
readContacts();
}
}
// 获取联系人
private void readContacts(){ Cursor cursor = null;
try{
cursor = getContentResolver().query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI,null,null,null,null);
if (cursor != null){ while(cursor.moveToNext()){
// 联系人的名称
String displayName = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
// 联系人的手机号
String number = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
list.add(displayName+"\n"+number);
}
// 给适配器发消息说数据改变了,这时候会重新刷新一次数据
adapter.notifyDataSetChanged();
} }catch (Exception e){
e.printStackTrace();
}finally {
if (cursor != null){
cursor.close();
}
}
} // 获取权限结果
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults); switch (requestCode){
case 2:
if (grantResults.length>0&& grantResults[0] == PackageManager.PERMISSION_GRANTED){
readContacts();
}else {
Toast.makeText(this,"访问通讯录需要权限",Toast.LENGTH_SHORT).show();
}
}
}
}

创建自己的内容提供器


前面也说过,要是想实现跨程序之间的共享数据的功能,官方推荐的就是使用内容提供器,我们可以新建一个类去继承ContentProvider的方式来创建一个自己的内容提供器。ContentProvider里面的6个抽象方法我们全部重写

onCreate()

初始化内容提供器的时候调用,通常会在这里完成数据库的创建和升级操作,返回true表示成功,返回false表示失败,注意的是只有当ContentResolver尝试访问我们的数据库的时候内容提供器才会被初始化

query()

从内容提供器中查询数据,使用uri参数来确定查询哪张表,具体的参数我们就不在说了,前面我们已经说过,查询的具体的结果就在cursor对象中存放返回

insert()

添加数据我们也就不再说了,成功之后会返回一个用于表示这条记录的URI

update()

注意的就一点,受影响的函数将作为返回值返回

delete()

这个和更新一样也是将受影响的行数作为返回值返回

getType()

根据传入的内容URI来返回相应的MIME类型

方法具体的内容我们就不在多说了,可以自己点进类里面去看看。有一点需要注意的就是URI,有一点需要我们注意:

*  表示匹配任意长度的任意字符

#  表示匹配任意长度的数字

所以,我们把一个能够匹配任意表的内容URI可以写成:content://com.example.app.provider/*

               我们把一个能够匹配表中任意一行数据的内容URI可以写成:content://com.example.app.provider/table1/#

最后还有一个问题,就是内容URI的匹配问题,有个类UriMatcher类可以了解一下,我们就不在多说了,想想你匹配好了URI,当外面用户进行CRUD操作的时候具体的返回内容就是我们自己自定义的了!以上就是我们要说的关于内容控制器的内容!

Android学习--跨程序共享数据之内容提供其探究的更多相关文章

  1. android——实现跨程序访问数据

    使用之前的SQLite存储的应用程序.首先需要在这个应用程序中创建内容提供器,右击com.example.administrator.exp7包→New→Other→Content Provider, ...

  2. Android学习笔记(十九)——内容提供器

    //此系列博文是<第一行Android代码>的学习笔记,如有错漏,欢迎指正! 内容提供器(Content Provider)主要用于在不同的应用程序之间实现数据共享的功能,它提供了一套完整 ...

  3. 内容提供者(Content Provider)——跨程序共享数据

    内容提供者 Content Provider 应用的数据库是不允许其他应用访问的 内容提供者的作用就是让别的应用访问到你的数据库 自定义内容提供者,继承ContentProvider类,重写增删改查方 ...

  4. 5、Android-跨程序共享数据--内容提供器

    Android数据持久化技术:文件存储.SharedPreferences存储.数据库存储 使用这些持久化技术保存的数据只能再当前的应用程序中访问 但是对于不同应用之间的可以实现跨程序数据共享的功能 ...

  5. android: 实现跨程序数据共享

    简单起见,我们还是在上一章中 DatabaseTest 项目的基础上继续开发,通过内容提供器 来给它加入外部访问接口. 打开 DatabaseTest 项目,首先将 MyDatabaseHelper ...

  6. Android入门(十四)内容提供器-实现跨程序共享实例

    原文链接:http://www.orlion.ga/661/ 打开SQLite博文中创建的 DatabaseDemo项目,首先将 MyDatabaseHelper中使用 Toast弹出创建数据库成功的 ...

  7. 基于Ceph分布式集群实现docker跨主机共享数据卷

    上篇文章介绍了如何使用docker部署Ceph分布式存储集群,本篇在此基础之上,介绍如何基于Ceph分布式存储集群实现docker跨主机共享数据卷. 1.环境准备 在原来的环境基础之上,新增一台cen ...

  8. Android学习笔记之JSON数据解析

    转载:Android学习笔记44:JSON数据解析 JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,采用完全独立于语言的文本格式,为Web应用开发提供了一种 ...

  9. android学习经常使用的数据文件夹

    android工程实践 1.仿360一键清理实现(一) "一键清理"是一个桌面图标,点击图标后,显示一个视图.进行清理动画.之后显示清理了几个进程,释放了多少M内存.点击" ...

随机推荐

  1. Luogu2737 USACO4.1麦香牛块(动态规划)

    小凯的疑惑升级版.只有两个数的话不能表示的最大数是ab-a-b,显然如果可选数增加不会比这更大,所以只要答案存在一定小于256*256-2*256.在这个范围内背包即可. #include<io ...

  2. hdu6097 Mindis(几何)

    题解: 这里是用解析解的做法, 我们发现如果以P和Q做椭圆,那么当椭圆与圆相切的时候,答案最优 那么方程就是这样的 联立之后,解delta等于0,可以得到 答案就是2a了 注意不一定任何情况都有解,当 ...

  3. 【BZOJ 2553】[BeiJing2011]禁忌 AC自动机+期望概率dp

    我一开始想的是倒着来,发现太屎,后来想到了一种神奇的方法——我们带着一个既有期望又有概率的矩阵,偶数(2*id)代表期望,奇数(2*id+1)代表概率,初始答案矩阵一列,1的位置为1(起点为0),工具 ...

  4. 【NOIP 模拟赛】中值滤波 打表找规律

    对于这样看起来不像什么算法也没什么知识点的题,一脸懵逼的话不是手推规律就是打表找规律......... 当然还有一些超出你能力之外的数学题...... #include <cstdio> ...

  5. bzoj5091 [Lydsy1711月赛]摘苹果 概率题

    [Lydsy1711月赛]摘苹果 Time Limit: 1 Sec  Memory Limit: 256 MBSubmit: 174  Solved: 135[Submit][Status][Dis ...

  6. org.apache.hadoop.hdfs.server.datanode.DataNode: Exception in receiveBlock for block

    Hbase依赖的datanode日志中如果出现如下报错信息:DataXceiverjava.io.EOFException: INFO org.apache.hadoop.hdfs.server.da ...

  7. SQLNET跟踪tnsping过程

    原文地址:SQLNET跟踪tnsping过程 作者:yingyifeng306 sqlnet是oracle提供的与网络层面交互的一个工具,比如如何解析客户端发起的连接,如何对客户端发起的连接进行辨别, ...

  8. jquery中:input和input的区别

    :input表示选择表单中的input,select,textarea,button元素, input仅仅选择input元素. <button>和<input type=" ...

  9. bzoj3223 文艺平衡树 codevs3303 翻转区间

    splay模版题吧 只有区间翻转 至于为什么要把须翻转区间旋到根 因为查找一个区间可以先找出他左端点左边第一个点和右端点x右边第一个点y 然后将x旋到根节点 y旋到x的右儿子 这样x的右边的点就是所有 ...

  10. Shell中的while循环【转】

    转自:http://blog.chinaunix.net/uid-25880122-id-2901409.html while循环的格式   while expression do command c ...