安卓开发 探究内容提供器

  • 前言: 只是我个人在阅读一行代码时,写的一些笔记,主要是敲一遍代码,然后结合我的理解复述下,有哪里不对的,请大佬指点。

1. 内容提供器简介

实际上我个人的理解就是内容提供器是为了让跨进程读取更加规范和安全,之前使用的sharePreferences和文件存储曾经也有提供几种模式提供外部的应用程序可以进行读取,但是并不安全,所以安卓官方已经将这种方式废除了,采取了更加规范安全的内容提供器技术

2. 运行时权限

这个和我们生活就比较密切了,有时候我们使用qq,想拍照发送图片,会提示,是否使用相机, 其实这个运行时权限,在安卓6之前并没有出现,记得小时候安装个app,就会弹出各种权限的申请,不安装就不让你使用,非常流氓,所以安卓官方在安卓6时就推出了运行时权限申请,让用户在安装app时可以不全部同意权限申请,也可以使用app的功能,只需要在用户使用到权限功能时,弹出对话框,让用户手动授权,不过实际上权限也是分为普通权限和危险权限,普通权限系统是自动授权的,而危险权限系统必须用户手动授权。

3. 运行时权限代码案例:

package com.example.contentprovidertest;

import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;

import android.Manifest;
import android.app.Activity;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.ExpandableListAdapter;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button makeCall=(Button)findViewById(R.id.make_call);
        makeCall.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
               if(ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.CALL_PHONE)!= PackageManager.PERMISSION_GRANTED)
               {
                   ActivityCompat.requestPermissions(MainActivity.this,new String[]{Manifest.permission.CALL_PHONE},1);
               }else
               {
                   call();
               }
            }
        });
    }
    private void call()
    {
        try{
            Intent intent=new Intent(Intent.ACTION_CALL);
            intent.setData(Uri.parse("tel:10086"));
            startActivity(intent);
        }catch (Exception e)
        {
            e.printStackTrace();
        }
    }
    public void onRequestPermissionsResult(int requestsCode,String[] permissions,int[] grantResult)
    {
        switch (requestsCode)
        {
            case 1:
                if(grantResult.length>0&&grantResult[0]==PackageManager.PERMISSION_GRANTED)
                {
                    call();
                }else{
                    Toast.makeText(this,"You denied the permission",Toast.LENGTH_SHORT).show();
                }
                break;
            default:

        }

    }
}

哦对,必须在AndroidManifest.xml中声明权限

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.contentprovidertest">
    <uses-permission android:name="android.permission.CALL_PHONE"/>

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/Theme.ContentProviderTest">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

其实代码也不难理解,就是使用ContextCompat.checkSelfPermission方法,检查权限是否已经授权过了,授权了就直接调用就好了,
如果没有,就主动申请,ActivityCompat.requestPermissions方法主动申请,弹出对话框,让你选择,点击同意或拒绝,都会调用
onRequestPermissionsResult方法

4. 访问其他程序的数据

内容提供器实际上就是个接口,我们可以利用内容提供器对外部应用程序的数据进行读取,也可以自己自定义内容提供器向外部提供自己的数据。
有些固定用法,我直接copy书上了,用法和SQLite差别不大,不过注意的点在于,能够定位位置的点,不止表名了,因为单一个表名无法定位是
在哪个app中进行读取,所以又加入了包名,包名加表名,就安排上了


固定格式,直接照书上格式弄呗

5. 读取系统联系人

package com.example.contentprovidertest;

import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;

import android.Manifest;
import android.app.Activity;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.provider.ContactsContract;
import android.view.View;
import android.widget.ArrayAdapter;
import android.widget.Button;
import android.widget.ExpandableListAdapter;
import android.widget.ListView;
import android.widget.Toast;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity {
    ArrayAdapter<String> adapter;

    List<String> contactList=new ArrayList<>();
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ListView contactsView=(ListView)findViewById(R.id.contacts_view);
        adapter=new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1,contactList);
        contactsView.setAdapter(adapter);
        if(ContextCompat.checkSelfPermission(this,Manifest.permission.READ_CONTACTS)!=PackageManager.PERMISSION_GRANTED)
        {
            ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.READ_CONTACTS},1);
        }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));
                    contactList.add(displayName +"\n"+number);
                }
                adapter.notifyDataSetChanged();
            }
        }catch (Exception e)
        {
            e.printStackTrace();
        }finally {
            if(cursor!=null)
            {
                cursor.close();
            }
        }
    }
    public void onRequestPermissionsResult(int requestCode,String[] permissions,int[] grantResults)
    {
        switch (requestCode)
        {
            case 1:
                if(grantResults.length>0&&grantResults[0]==PackageManager.PERMISSION_GRANTED)
                {
                    readContacts();
                }else
                {
                    Toast.makeText(this,"You denied the permission",Toast.LENGTH_SHORT).show();
                }
                break;
            default:
        }
    }


}
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.contentprovidertest">
    <uses-permission android:name="android.permission.CALL_PHONE"/>
    <uses-permission android:name="android.permission.READ_CONTACTS"/>
    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/Theme.ContentProviderTest">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

因为读取联系人是危险权限,所以我们需要运行时权限,让用户手动授权,ContactsContract.CommonDataKinds.Phone.CONTENT_UR是安卓封装好的
uri字符串,然后我们对其进行读取,让内容放入list中,再更新listview好了
6. 自定义内容提供器

posted @ 2021-02-07 12:16  YenKoc  阅读(127)  评论(0编辑  收藏  举报