Androidc學習筆記五之四大組件之內容提供器ContentProvider


android的內容提供器使用方法 android6.0以上獲取運行是權限方法   本地內容提供器和自定義內容提供器

2017/3/10  記載

《內容提供器》

內容提供器:主要是用於在不同應用程序之間共享數據的,打個比如,很多應用需要獲取本地系統中的
電話簿GPRS定位、還有其他應用的數據,這些都是需要共享的。

運行時權限:在6.0android以下的版本,都是在安裝應用的時候一次性獲取所有的權限但是在6.0之后
谷歌考慮到安全性的問題和實用性的問題,就拋棄了以前的方式,采用在應用運行的時候要用到的時候,提示獲取
權限
,若用戶不同意就不能用,但是依然可以使用其他功能,從而避免了店大欺客的問題。

Android將現在的權限分為普通權限危險權限,普通權限是不會威脅到用戶的數據安全性問題。
就可以在安裝時候允許或者由系統自動給你判定。危險權限就需要用戶自行判斷了,危險權限也不多如下所示:

效果圖:

                                  


注意:Android系統為我們提供了很多服務管理的類,包括ActivityManager、PowerManager(電源管理)、AudioManager(音頻管理)
等。除此之外,還提供了一個PackageManger(需要了解一下)管理類,它的主要職責是管理應用程序包。 通過它,我們就可以獲取應用程序信息。

做一個在運行時可以獲取的權限例子RuntimePermissionProject:當點擊按鈕撥打電話10086,這個時候就會調用運行時權限了。
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">

<Button
android:id="@+id/make_call"
android:text="Make Call"
android:textAllCaps="false"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
</LinearLayout>

AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.ldp.com.runtimepermissionproject">

<!--添加權限-->
<uses-permission android:name="android.permission.CALL_PHONE"/>
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<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>

MainActivity
package com.example.ldp.com.runtimepermissionproject;


import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;


public class MainActivity extends AppCompatActivity {


@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);


/*點擊按鈕要獲取權限*/
Button buttonCall = (Button)findViewById(R.id.make_call);
buttonCall.setOnClickListener(new View.OnClickListener() {
/**ContextCompat.checkSelfPermission()檢查用戶是否已經給我們授權了(第二個參數是什么權限)。
* PackageManager調用應用權限組比較,給了就直接執行Call(),如果沒有給權限通過
* ActivityCompat.requestPermissions(Activity的實例,申請的權限放在其中,請求碼(唯一值)這里設置為1)方法請求權限
* */


@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 (SecurityException e){
e.printStackTrace();
}
}


/*不論用戶同意還是拒絕,
*activity的onRequestPermissionsResult會被回調來通知結果(通過第三個參數),grantResults,如下*/
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
switch (requestCode){
case 1:
if (grantResults.length>0&&grantResults[0]== PackageManager.PERMISSION_GRANTED){
call();
}else{
Toast.makeText(this,"You Denied the permission",Toast.LENGTH_SHORT).show();
}
break;
default:
}
}
}

效果圖:因為我這里使用的是夜神模擬器,所以出現不支持的情況,不過代碼沒問題,可以使用本機模擬器試一下
                              

進入正題了,內容提供器 ContentResolver
1.使用現有的內容提空氣來讀取相應程序中的數據
2.自己創建一個內容提供器來給應用提供數據外部訪問接口


說明:
效果圖:


      


寫一個例子來通過ContentResolver來獲取本地電話簿
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">

<ListView
android:id="@+id/contacts_view"
android:layout_width="match_parent"
android:layout_height="match_parent">
</ListView>
</LinearLayout>


AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.ldp.com.contenttest">


<uses-permission android:name="android.permission.READ_CONTACTS"/>
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<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>


MainActivity
package com.example.ldp.com.contenttest;


import android.Manifest;
import android.content.pm.PackageManager;
import android.database.Cursor;
import android.provider.ContactsContract;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.Switch;
import android.widget.Toast;


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


public class MainActivity extends AppCompatActivity {


ArrayAdapter<String> adapter;
List<String> contractsList = new ArrayList<>();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);


ListView constactView = (ListView) findViewById(R.id.contacts_view);
adapter = new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1,contractsList);
constactView.setAdapter(adapter);
if(ContextCompat.checkSelfPermission(this, Manifest.permission.READ_CONTACTS)!= PackageManager.PERMISSION_GRANTED){
ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.READ_CONTACTS},1);
}else{


}
}


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));
contractsList.add(displayName+"\n"+number);
}
adapter.notifyDataSetChanged();
}


}catch (Exception e){
e.printStackTrace();
}finally {
if(cursor!=null){
cursor.close();
}
}
}


@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull 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:
}
}
}

兩張效果圖:

獲取權限:

                          

允許權限:

                        



上,面是使用本地內置的內容提供器的例子,接下來使用直接定義的內容提供器
直接創建內容提供器:
想要使用擴程序的共享數據功能,推薦使用繼承ContentProvider(6個抽象方法)的方式創建

參數效果圖:

            


利用UriMatcher中的addURI(authority,Path,自定義代碼),可以講一個Uri對象傳入返回值是某個能夠匹配
這個Uri對象所對應的自定義代碼。
效果圖:
      






      






注意!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系我们删除。



 
粤ICP备14056181号  © 2014-2021 ITdaan.com