Android獲取設(shè)備隱私 忽略6.0權(quán)限管理
一.前言
(1).由于MIUI等部分國產(chǎn)定制系統(tǒng)也有權(quán)限管理,沒有相關(guān)api,故無法判斷用戶是否允許獲取聯(lián)系人等隱私。在Android 6.0之后,新增權(quán)限管理可以通過官方api判斷用戶的運行狀態(tài);
(2).我們指定targetSdkVersion為23或者之后我們還需要在運行時請求這些所需的權(quán)限。這很重要,因為已經(jīng)出現(xiàn)了很多開發(fā)者把targetSdkVersion飆到了最新,然后發(fā)現(xiàn)自己的app瘋狂的崩潰,這是由于他們沒有實現(xiàn)執(zhí)行運行時權(quán)限請求的代碼。當你已經(jīng)把一個targeting API 為23或者之后的app發(fā)布到了Google Play上,這更是一個問題,你無法立即把那個apk的targeting API替換成更早的版本。
二.權(quán)限分析
從Android6.0開始,權(quán)限分為普通權(quán)限和許可權(quán)限。許可權(quán)限分類歸組,一個權(quán)限授權(quán)之后,該組下的權(quán)限均可使用。
(1)普通權(quán)限
只需要在xml申請即可,使用方法和之前6.0以前的一樣。在應(yīng)用安裝應(yīng)用時,會默認獲得許可。
(2)許可權(quán)限
可執(zhí)行 $adb shell pm list permissions -d -g

同一組的任何一個權(quán)限被授權(quán)了,其他權(quán)限也自動被授權(quán)。例如,一旦WRITE_CONTACTS被授權(quán)了,app也有READ_CONTACTS和GET_ACCOUNTS了。
源碼中被用來檢查和請求權(quán)限的方法分別是Activity的checkSelfPermission和requestPermissions,這些方法api23引入。
三.相關(guān)方法
(1).ContextCompat.checkSelfPermission()
檢查應(yīng)用是否擁有該權(quán)限,被授權(quán)返回值為PERMISSION_GRANTED,否則返回PERMISSION_DENIED
(2).ActivityCompat.requestPermissions()
將彈出請求授權(quán)對話框,這個方法在M之前版本調(diào)用,OnRequestPermissionsResultCallback 直接被調(diào)用,帶著正確的 PERMISSION_GRANTED或者 PERMISSION_DENIED 。
(3).AppCompatActivity.onRequestPermissionsResult()
該方法類似于Activity的OnActivityResult()的回調(diào)方法,主要接收請求授權(quán)的返回值
//版本判斷
if (Build.VERSION.SDK_INT >= 23) {
//減少是否擁有權(quán)限
int checkCallPhonePermission = ContextCompat.checkSelfPermission(getApplicationContext(), permission);
if (checkCallPhonePermission != PackageManager.PERMISSION_GRANTED) {
//彈出對話框接收權(quán)限
ActivityCompat.requestPermissions(BaseActivity.this, new String[]{permission}, id);
return;
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
//TODO:已授權(quán)
} else {
//TODO:用戶拒絕
}
}
(4).封裝
public class BaseActivity extends AppCompatActivity {
private Map<Integer, Runnable> allowablePermissionRunnables = new HashMap<>();
private Map<Integer, Runnable> disallowablePermissionRunnables = new HashMap<>();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
/**
* 請求權(quán)限
* @param id 請求授權(quán)的id 唯一標識即可
* @param permission 請求的權(quán)限
* @param allowableRunnable 同意授權(quán)后的操作
* @param disallowableRunnable 禁止權(quán)限后的操作
*/
protected void requestPermission(int id, String permission, Runnable allowableRunnable, Runnable disallowableRunnable) {
if (allowableRunnable == null) {
throw new IllegalArgumentException("allowableRunnable == null");
}
allowablePermissionRunnables.put(id, allowableRunnable);
if (disallowableRunnable != null) {
disallowablePermissionRunnables.put(id, disallowableRunnable);
}
//版本判斷
if (Build.VERSION.SDK_INT >= 23) {
//減少是否擁有權(quán)限
int checkCallPhonePermission = ContextCompat.checkSelfPermission(getApplicationContext(), permission);
if (checkCallPhonePermission != PackageManager.PERMISSION_GRANTED) {
//彈出對話框接收權(quán)限
ActivityCompat.requestPermissions(BaseActivity.this, new String[]{permission}, id);
return;
} else {
allowableRunnable.run();
}
} else {
allowableRunnable.run();
}
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
Runnable allowRun = allowablePermissionRunnables.get(requestCode);
allowRun.run();
} else {
Runnable disallowRun = disallowablePermissionRunnables.get(requestCode);
disallowRun.run();
}
}
}
public class MainActivity extends BaseActivity implements View.OnClickListener{
private Button btCallPhone;
private Button btContact;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btCallPhone = (Button) findViewById(R.id.call_phone);
btContact = (Button) findViewById(R.id.contact);
btCallPhone.setOnClickListener(this);
btContact.setOnClickListener(this);
}
@Override
public void onClick(View v) {
if(v == btCallPhone){
//撥打電話
requestPermission(1, Manifest.permission.CALL_PHONE, new Runnable() {
@Override
public void run() {
callPhone();
}
}, new Runnable() {
@Override
public void run() {
callPhoneDenied();
}
});
}else if(v == btContact){
//讀取聯(lián)系人信息
requestPermission(2, Manifest.permission.WRITE_CONTACTS, new Runnable() {
@Override
public void run() {
readContact();
}
}, new Runnable() {
@Override
public void run() {
readContactDenied();
}
});
}
}
private void callPhone() {
Toast.makeText(MainActivity.this, "CALL_PHONE OK", Toast.LENGTH_SHORT)
.show();
}
private void callPhoneDenied() {
Toast.makeText(MainActivity.this, "CALL_PHONE Denied", Toast.LENGTH_SHORT)
.show();
}
private void readContact() {
ContentResolver cr = getContentResolver();
String str[] = {ContactsContract.CommonDataKinds.Phone.CONTACT_ID, ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME, ContactsContract.CommonDataKinds.Phone.NUMBER,
ContactsContract.CommonDataKinds.Phone.PHOTO_ID};
Cursor cur = cr.query(
ContactsContract.CommonDataKinds.Phone.CONTENT_URI, str, null,
null, null);
int count = cur.getCount();
cur.close();
Toast.makeText(MainActivity.this, String.format("發(fā)現(xiàn)%s條", count), Toast.LENGTH_SHORT)
.show();
}
private void readContactDenied() {
Toast.makeText(MainActivity.this, "Contact Denied", Toast.LENGTH_SHORT)
.show();
}
}
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助。
相關(guān)文章
詳解Android studio 動態(tài)fragment的用法
這篇文章主要介紹了Android studio 動態(tài)fragment的用法,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-10-10
Android編程Widget創(chuàng)建與使用方法簡明教程
這篇文章主要介紹了Android編程Widget創(chuàng)建與使用方法,結(jié)合實例形式分析了Widget的功能、使用方法與相關(guān)注意事項,需要的朋友可以參考下2016-10-10
Android中Permission權(quán)限機制的具體使用
這篇文章主要介紹了Android中Permission權(quán)限機制的具體使用,本文講解了權(quán)限級別 protection level、ICC(inter-component communication)權(quán)限保護等內(nèi)容,需要的朋友可以參考下2015-04-04
Android程序開發(fā)通過HttpURLConnection上傳文件到服務(wù)器
這篇文章主要介紹了Android程序開發(fā)通過HttpURLConnection上傳文件到服務(wù)器的相關(guān)資料,需要的朋友可以參考下2016-01-01
Android神兵利器之Image Asset Studio的實現(xiàn)
這篇文章主要介紹了Android神兵利器之Image Asset Studio的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-06-06
淺談關(guān)于Android WebView上傳文件的解決方案
這篇文章主要介紹了淺談關(guān)于Android WebView上傳文件的解決方案 ,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-09-09
Android launcher中模擬按home鍵的實現(xiàn)
這篇文章主要介紹了Android launcher中模擬按home鍵的實現(xiàn)的相關(guān)資料,需要的朋友可以參考下2017-05-05

