安信科技官网-触屏版
当前位置:首页 > 安信自媒体 > 移动微信

Android开发中使用FileProvider解决apk升级包无法安装的问题

时间:2023年11月29日 浏览:

Android使用FileProvider解决apk无法安装的问题

安卓开发APP,需要提供APP升级包安装时,提示出现下面的问题:

W/System.err: android.os.FileUriExposedException: file:///xxxx/xxxx/xxxx/updata.apk exposed beyond app through Intent.getData()

上面的错误代码中:updata.apk为升级包。
通用使用下面的方法解决处理:

1.配置mainfest信息,在mainfest的application结点内添加如下代码:
android:name="androidx.core.content.FileProvider"
android:authorities="com.anxin.myapp.fileprovider"
android:exported="false"
android:grantUriPermissions="true"
tools:ignore="WrongManifestParent">
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/filepaths" />


上面的代码中:
android:authorities 这里使用的是包名。这个其实可以随便写,换成自己的包名就都可以。
android:resource 这里配置一个 xml文件。命名为:filepaths.

2.在xml下创建filepaths.xml文件如下:
<paths>
    <external-path name="." path="."/>
</paths>


以下代码说明:
external-path 是放到sd卡的目录下 Environment.getExternalStorageDirectory()
path="." 代表共享 sd卡下的所有目录。会遍历sd卡下的所有目录,来匹配你要安装 的apk 的目录。。
name="." 代表apk存放目录下所有apk的名字都会遍历一遍,然后跟你要安装的apk进行匹配

3.以下是判断远程升级包版本号,并下载、安装升级包APP的代码:

public class Updateapk extends BaseData {
Context context;
private RadioGroup mRg;
ImageView iv_back;
String url="http://远程服务器/getjson/getnewversion";
TextView tv_versionName,tv_versioninfo;
Button btn_backusercenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.updateapk);
initRadioGroup();
findViewById();
loadData(url);
}
private void findViewById() {
iv_back=findViewById(R.id.iv_back);
iv_back.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
finish();
}
});
btn_backusercenter=findViewById(R.id.btn_backusercenter);
btn_backusercenter.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
finish();
}
});
}
@Override
public void onSuccess(String result) {
try {
parseShowData(result);
} catch (PackageManager.NameNotFoundException e) {
e.printStackTrace();
}
}

private void parseShowData(String result) throws PackageManager.NameNotFoundException {
BeanUpdate bean=new Gson().fromJson(result,BeanUpdate.class);
Integer serviceVersionCode= bean.getServiceversionCode();
String serviceversionName=bean.getServiceversionName();
String serviceversionInfo=bean.getServiceversionInfo();
tv_versionName=findViewById(R.id.tv_versionName);
tv_versionName.setText("新版本待升级,新版本号:"+serviceversionName);
tv_versioninfo=findViewById(R.id.tv_versioninfo);
tv_versioninfo.setText(serviceversionInfo);
if (getVersionCode() < serviceVersionCode) {
showDialogUpdate();

}else{
Toast.makeText(this,"当前已经是最新的版本", Toast.LENGTH_SHORT).show();
}
}
private void showDialogUpdate() {
AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle("版本升级").
setIcon(R.mipmap.ic_launcher).
setMessage("发现新版本!请及时更新").
setPositiveButton("确定", new DialogInterface.OnClickListener() {

@Override
public void onClick(DialogInterface dialog, int which) {
loadNewVersionProgress();//下载最新的版本程序
}
}).
setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Intent intent=new Intent();
intent.setClass(Updateapk.this, UserCenter.class);
startActivity(intent);
}
});
AlertDialog alertDialog = builder.create();
alertDialog.show();
}
/**
* 下载新版本程序,需要子线程
*/
private void loadNewVersionProgress() {
final String uri="http://xxxx.xxxx.com/updata.apk";
final ProgressDialog pd;//进度条对话框
pd = new  ProgressDialog(this);
pd.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
pd.setMessage("正在下载更新");
pd.show();
//启动子线程下载任务
new Thread(){
@Override
public void run() {
try {
File file = getFileFromServer(uri, pd);
sleep(3000);
installApk(file);
pd.dismiss();
} catch (Exception e) {
Toast.makeText(getApplicationContext(), "下载新版本失败", Toast.LENGTH_LONG).show();
e.printStackTrace();
}
}}.start();
}
/**
* 从服务器获取apk文件的代码
* 传入网址uri,进度条对象即可获得一个File文件(要在子线程中执行哦)
*/
public static File getFileFromServer(String uri, ProgressDialog pd) throws Exception{
if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){
URL url = new URL(uri);
HttpURLConnection conn =  (HttpURLConnection) url.openConnection();
conn.setConnectTimeout(5000);
pd.setMax(conn.getContentLength());
InputStream is = conn.getInputStream();
long time= System.currentTimeMillis();
File file = new File(Environment.getExternalStorageDirectory()+"/Download/", time+"updata.apk");
FileOutputStream fos = new FileOutputStream(file);
BufferedInputStream bis = new BufferedInputStream(is);
byte[] buffer = new byte[1024];
int len ;
int total=0;
while((len =bis.read(buffer))!=-1){
fos.write(buffer, 0, len);
total+= len;
pd.setProgress(total);
}
fos.close();
bis.close();
is.close();
return file;
}
else{
return null;
}
}

/**
* 安装apk
*
* @param file
*/
private void installApk(File file) {
//File file = new File(fileSavePath);
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
Uri data;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {//判断版本大于等于7.0
// "com.anxin.loveenglish.fileprovider"即是在清单文件中配置的authorities
// 通过FileProvider创建一个content类型的Uri
data = FileProvider.getUriForFile(this, "com.anxin.loveenglish.fileprovider", file);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);// 给目标应用一个临时授权
} else {
data = Uri.fromFile(file);
}
intent.setDataAndType(data, "application/vnd.android.package-archive");
startActivity(intent);
}

@Override
public void onError(Throwable ex, boolean isOnCallback) {
Toast toast =Toast.makeText(this, "当前为最新版本,不需要升级!", Toast.LENGTH_SHORT);
toast.setGravity(Gravity.CENTER, 0, 0);
toast.show();
Intent intent=new Intent();
intent.setClass(Updateapk.this, UserCenter.class);
startActivity(intent);
}

/*获取当前程序的版本号*/

private int getVersionCode() throws PackageManager.NameNotFoundException {
//获取packagemanager的实例
PackageManager packageManager = getPackageManager();
//getPackageName()是你当前类的包名,0代表是获取版本信息
PackageInfo packInfo = packageManager.getPackageInfo(getPackageName(), 0);
return packInfo.versionCode;
}
}


最后说明下,http://远程服务器/getjson/getnewversion

这里获取的是远程服务器中设置的APP新版本信息,以tp6.0为例:

class Getjson extends Base{
public function getnewversion(){
$data['serviceversionName']='1.2.0';
$data['serviceversionCode']=2;
$data['serviceversionInfo']='最新版本说明文字';
return json($data);
}

上面的代码:serviceversionCode使用自然数,不要使用带小数点。


当前页面电脑端访问网址:


发送给朋友 分享到朋友圈

手机版电脑版