项目中发现在自动更新功能的时候,下载好了apk的文件后在android7.0系统中不能自动跳到安装界面,后来搜索了一番解决了问题,但感觉没有描述清楚,所以补充一下。
info: 与上一篇配合使用,最近在我们的应用中加入更新功能,按照往常一样加入代码
1
2
3
4
5
6
7
8
|
if
(!apkfile.exists()) {
Toast.makeText(mContext,
"下载的安装包不存在"
, Toast.LENGTH_SHORT).show();
return
;
}
Intent install =
new
Intent(Intent.ACTION_VIEW);
install.setDataAndType(Uri.fromFile(apkfile),
"application/vnd.android.package-archive"
);
install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
mContext.startActivity(install);
|
在测试机上一运行,妥妥的能安装,就此完事,然后我把它打包后发到我自己的手机上(Android7.1.1)上,下载安装,居然报错了,然后我的同事就说你看看7.0以上的文件,好像加了限制了,于是上百度,果然是这样的,谷歌在Android7.0引入私有目录被限制访问和StrictMode API。
那么我们在7.0以上怎么适配呢?别急,谷歌给我们提供了方法就是使用FileProvider。知道这个解决方法之后我们就开始写代码吧。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
if
(!apkfile.exists()) {
Toast.makeText(mContext,
"下载的安装包不存在"
, Toast.LENGTH_SHORT).show();
return
;
}
//判读版本是否在7.0以上
if
(Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
//在AndroidManifest中的android:authorities值
Uri apkUri = FileProvider.getUriForFile(mContext,
"你的包名.provider"
, apkfile);
Intent install =
new
Intent(Intent.ACTION_VIEW);
install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
//添加这一句表示对目标应用临时授权该Uri所代表的文件
install.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
install.setDataAndType(apkUri,
"application/vnd.android.package-archive"
);
mContext.startActivity(install);
}
else
{
Intent install =
new
Intent(Intent.ACTION_VIEW);
install.setDataAndType(Uri.fromFile(apkfile),
"application/vnd.android.package-archive"
);
install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
mContext.startActivity(install);
}
|
然后在AndroidManifest.xml中配置
1
2
3
4
5
6
7
8
9
|
<provider
android:name=
".service.MyFileProvider"
android:authorities=
"你的包名.provider"
android:exported=
"false"
android:grantUriPermissions=
"true"
>
<meta-data
android:name=
"android.support.FILE_PROVIDER_PATHS"
android:resource=
"@xml/file_paths"
/>
</provider>
|
最后在res文件夹底下新建一个xml文件夹,里面新建file_paths.xml文件
1
2
3
4
5
6
|
<?
xml
version
=
"1.0"
encoding
=
"utf-8"
?>
<
resources
>
<
paths
>
<
external-path
path
=
""
name
=
"download"
/>
</
paths
>
</
resources
>
|
当这些都写完之后,别以为就此完事了,别忘了权限
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
<uses-permission android:name="android.permission.INTERNET"/>
最后,别忘了在代码中动态申请权限(6.0的坑)。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
if
(Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
//TODO 更新
}
else
{
if
(ContextCompat.checkSelfPermission(
this
, Manifest.permission.WRITE_EXTERNAL_STORAGE)
== PackageManager.PERMISSION_GRANTED) {
//TODO 更新
}
else
{
//申请权限
ActivityCompat.requestPermissions(
this
,
new
String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
1
);
}
}
//权限请求结果
@Override
public
void
onRequestPermissionsResult(
int
requestCode,
@NonNull
String[] permissions,
@NonNull
int
[] grantResults) {
super
.onRequestPermissionsResult(requestCode, permissions, grantResults);
switch
(requestCode) {
case
1
:
if
(grantResults[
0
] == PackageManager.PERMISSION_GRANTED) {
//TODO 更新
}
else
{
Toast.makeText(
this
,
"暂无读写SD卡权限"
, Toast.LENGTH_SHORT).show();
}
break
;
}
}
|
当完成以上这些代码,我们就可以愉快的去更新你的应用了。