完美兼容7.0手機(jī)app升級(jí)問(wèn)題

大家好,一直想著寫點(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、是在寫入文件的時(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)的是所有路徑,而文件寫入時(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ì)文件寫入模塊的代碼修改

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



最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容