其中部分参照 网络大神的文章,自己整理了一下
android 8.0 系统,app内部升级失败,因为,Android 8.0 系统内部升级时需要 申请 安装权限 具体步骤如下:
1. 在 AndroidManifest 中 添加 以下权限 <uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" />
2. 在build.gradle 中 compileSdkVersion 与 targetSdkVersion 设置 26 即 8.0
compileSdkVersion 26
buildToolsVersion "26.0.3"
defaultConfig {
applicationId "com.syd.activity"
minSdkVersion 15
targetSdkVersion 26
versionCode 8
versionName "3.32"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
multiDexEnabled true
}
3. 在下载完成后,检查是否允许 安装未知应用
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
if(mContext.getPackageManager().canRequestPackageInstalls()){
// ToastUtil.showToast(mContext,"有权限,直接安装");
((InComeActivity) activity).installApk(file);
}else{
// ToastUtil.showToast(mContext,"没有权限,去申请跳转");
(InComeActivity) activity).installPower();
}
}
4. 若没有权限去申请跳转
//8.0跳转允许安装未知应用权限
public void installPower() {
//ToastUtil.showToast(this, "跳转");
Uri packageURI = Uri.parse("package:" + getPackageName());
Intent intent = new Intent(Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES, packageURI);
startActivityForResult(intent, 1122);
}
5. 在页面返回时再次校验是否有权限,有就直接安装
@RequiresApi(api = 26)
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == 1122) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
// 同意以后 进行安装
if (this.getPackageManager().canRequestPackageInstalls()) {
//ToastUtil.showToast(this, "有权限");
File apkfile = new File(saveFileName);
installApk(apkfile);
}
}
}
}
6. 进行安装 以下两个Flags 要加,要不然,版本更新时会 出现 解析包错误!
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
/**
* 安装apk
*
* @param
*/
public void installApk(File apkFile) {
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
Uri contentUri =
FileProvider.getUriForFile(this,"包名.fileprovider", apkFile);
intent.setDataAndType(contentUri, "application/vnd.android.package-archive");
//添加这一句表示对目标应用临时授权该Uri所代表的文件
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
} else {
intent.setDataAndType(Uri.fromFile(apkFile), "application/vnd.android.package-archive");
}
if (this.getPackageManager().queryIntentActivities(intent, 0).size() > 0) {
this.startActivity(intent);
}
}
7. 针对 FileProvider 的一些设置
在 AndroidManifest中 添加:
<!-- FileProvider配置访问路径,适配7.0及其以上 -->
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="包名.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths"/>
</provider>
8. 在 res中创建 xml目录,并创建 file_paths.xml 文件
<?xml version="1.0" encoding="utf-8"?>
<paths>
<external-path path="Android/data/包名/" name="files_root" />
<external-path path="." name="external_storage_root" />
</paths>
注:几处的包名要一致,不然会报空指针