大家好村砂,一直想著寫(xiě)點(diǎn)什么來(lái)在記敘開(kāi)發(fā)中遇到的問(wèn)題和解決方案肥哎,激勵(lì)自己,分享給需要的小伙伴坐榆!
話說(shuō)自從google出來(lái)Android 7.0系統(tǒng)之后拴魄,我們公司的測(cè)試小伙伴就向我提出了7.0的bug,接下來(lái)就著手查閱了下問(wèn)題是出在哪里的席镀,該如何來(lái)解決呢匹中。
bug日志是長(zhǎng)這樣子的:
android.os.FileUriExposedException: file:///storage/emulated/0/ys_toutiao.apk exposed beyond app through Intent.getData()
at android.os.StrictMode.onFileUriExposed(StrictMode.java:1816)
at android.net.Uri.checkFileUriExposed(Uri.java:2350)
at android.content.Intent.prepareToLeaveProcess(Intent.java:9076)
at android.content.Intent.prepareToLeaveProcess(Intent.java:9037)
at android.app.Instrumentation.execStartActivity(Instrumentation.java:1530)
at android.app.Activity.startActivityForResult(Activity.java:4391)
at android.support.v4.app.BaseFragmentActivityJB.startActivityForResult(BaseFragmentActivityJB.java)
at android.support.v4.app.FragmentActivity.startActivityForResult(FragmentActivity.java)
at android.app.Activity.startActivityForResult(Activity.java:4335)
at android.support.v4.app.FragmentActivity.startActivityForResult(FragmentActivity.java)
at android.app.Activity.startActivity(Activity.java:4697)
at android.app.Activity.startActivity(Activity.java:4665)
at com.ijuyin.prints.news.utils.VersionUtils$1.handleMessage(VersionUtils.java)
at android.os.Handler.dispatchMessage(Handler.java:105)
at android.os.Looper.loop(Looper.java:156)
at android.app.ActivityThread.main(ActivityThread.java:6524)
at java.lang.reflect.Method.invoke(Method.java)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:941)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:831)
也就是說(shuō)是在什么情況下會(huì)出現(xiàn)呢?
1豪诲、是在調(diào)用系統(tǒng)相機(jī)的時(shí)候顶捷;
2、是在寫(xiě)入文件的時(shí)候屎篱, 在我這里是強(qiáng)制升級(jí)后下載完成安裝Apk時(shí)服赎;
關(guān)于FileUriExposedException 異常的描述是這樣子的:
針對(duì)于權(quán)限部分,Android7.0是google推出的對(duì)權(quán)限做了一個(gè)更新即不允許出現(xiàn)以file://的形式調(diào)用隱式APP系統(tǒng)交播,也就是說(shuō)以前呢重虑,Uri的獲取方式是以file://xxx的樣式來(lái),那么我們也就是通過(guò)Uri.fromFile()來(lái)獲取如今放在7.0及以上系統(tǒng)呢秦士,這樣子就不行啦缺厉;
如今的解決關(guān)鍵在哪里呢,需要在應(yīng)用間共享文件,也就是需要發(fā)送一項(xiàng)content://URI提针,并授予 URI 臨時(shí)訪問(wèn)權(quán)限命爬。進(jìn)行此授權(quán)的最簡(jiǎn)單方式是使用FileProvider類。
嗯的辐脖,F(xiàn)ileProvider
1遇骑、首先我們需要在AndroidManifest中的application下添加provider:
http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"package="com.ijuyin.prints.news">
在這里我們需要注意一下其中設(shè)置的各種屬性的含義:
authorities:是該項(xiàng)目的包名+providergrantUriPermissions:必須是true,表示授予 URI 臨時(shí)訪問(wèn)權(quán)限 exported:必須是false resource:中的@xml/file_paths是我們接下來(lái)要在資源文件目錄下添加的文件
2揖曾、在res目錄下新建一個(gè)xml文件夾,并且新建一個(gè)file_paths的xml文件(如下圖)
設(shè)置file_path路徑.png
3亥啦、打開(kāi)file_paths.xml文件添加如下內(nèi)容
需要注意的是:
path:需要臨時(shí)授權(quán)訪問(wèn)的路徑(.代表在相機(jī)調(diào)用時(shí)候訪問(wèn)的是所有路徑炭剪,而文件寫(xiě)入時(shí)訪問(wèn)的路徑是Android/data/com.ijuyin.prints.news/) name: 是你為設(shè)置的這個(gè)訪問(wèn)路徑起的名字
4、接下來(lái)便是修改適配Android 7.0及以上系統(tǒng)的代碼
第一部分是對(duì)于相機(jī)模塊的修改
/**
* Open camera
*/
privatevoidshowCameraAction() {
if(ContextCompat.checkSelfPermission(getContext(), Manifest.permission
.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
requestPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE,
getString(R.string.mis_permission_rationale_write_storage),
REQUEST_STORAGE_WRITE_ACCESS_PERMISSION);
}else{
Intent intent =newIntent(MediaStore.ACTION_IMAGE_CAPTURE);
if(intent.resolveActivity(getActivity().getPackageManager()) !=null) {
try{
mTmpFile = FileUtils.createTmpFile(getActivity());
}catch(IOException e) {
e.printStackTrace();
}
Uri imageUri;
if(mTmpFile !=null&& mTmpFile.exists()) {
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
String authority = getActivity().getPackageName() +".provider";
imageUri = FileProvider.getUriForFile(getActivity(), authority, mTmpFile);
}else{
imageUri = Uri.fromFile(mTmpFile);
}
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent, REQUEST_CAMERA);
}else{
Toast.makeText(getActivity(), R.string.mis_error_image_not_exist, Toast
.LENGTH_SHORT).show();
}
}else{
Toast.makeText(getActivity(), R.string.mis_msg_no_camera, Toast.LENGTH_SHORT)
.show();
}
}
}
第二部分是對(duì)文件寫(xiě)入模塊的代碼修改
privatestaticHandler mHandler =newHandler() {
@Override
publicvoidhandleMessage(Message msg) {
if(msg.what == DOWN_UPDATE) {
mProgress.setProgress(progress);
}elseif(msg.what == DOWN_OVER) {
if(null!= downloadDialog && downloadDialog.isShowing()) {
try{
downloadDialog.dismiss();
}catch(Exception e) {
e.printStackTrace();
}
}
File apkfile =newFile(apkPath);
if(!apkfile.exists()) {
return;
}
Intent intent =newIntent(Intent.ACTION_VIEW);
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
String authority = mContext.getPackageName() +".provider";
Uri contentUri = FileProvider.getUriForFile(mContext, authority, apkfile);
intent.setDataAndType(contentUri,"application/vnd.android.package-archive");
}else{
intent.setDataAndType(Uri.fromFile(apkfile),"application/vnd.android"+
".package-archive");
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
}
mContext.startActivity(intent);
}
}
};
其中最核心的部分是
配置provider.png
需要注意的地方是:首先我們對(duì)Android系統(tǒng)的型號(hào)做出判斷添加flags翔脱,表明我們要被授予什么樣的臨時(shí)權(quán)限 以前我們直接 Uri.fromFile(apkFile)構(gòu)建出一個(gè)Uri,現(xiàn)在我們使用FileProvider.getUriForFile(getActivity(), getActivity().getPackageName() + ".provider", mTmpFile); 其中g(shù)etActivity().getPackageName()指的是該項(xiàng)目的應(yīng)用包名(此處調(diào)用的是在fragment奴拦,所以使用的是getActivity())
通過(guò)以上4步的設(shè)置操作,就可以完全解決Android 7.0及以上權(quán)限問(wèn)題導(dǎo)致的崩潰問(wèn)題届吁。
類似問(wèn)題鏈接:http://blog.csdn.net/world_kun/article/details/74276973