DownloadManager使用(适配7.0,8.0)

简介

感觉距离上次写文章已经过去很久很久啦,主要是一直没找到什么合适写的内容,而且抽空研究了一下Android以外的技术,写文章的事情就落下了。
最近要做一个app在线更新的功能,本想着自己写会不会有点小麻烦,因为涉及到一些7.0的apk文件下载后的获取,8.0的安装未知应用的权限,通知栏的设置什么的。然后突然发现有自带的DownloadManager可以用,现在想说真香~。

8.0安装未知应用权限申请


8.0安装未知应用权限申请
if (Build.VERSION.SDK_INT >= 26) {
    boolean b = getPackageManager().canRequestPackageInstalls();
    if (b) {
        //下载的具体方法
        downloadAPK();
    } else {
        //申请权限
        Uri packageURI = Uri.parse("package:" + getPackageName());
        Intent intent = new Intent(Settings.ACTION_MANAGE_UNKNOWN_APP_SOURCES, packageURI);
        startActivityForResult(intent, REQUEST_CODE_UNKNOWN_APP);
    }
} else {
    downloadAPK();
}

申请权限之后回调

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    //安装位未知应用的权限申请成功
    if (requestCode == REQUEST_CODE_UNKNOWN_APP) {
        LogUtil.d(resultCode + "");
        downloadAPK();
    }
}

就是一个很常规的权限申请,downloadAPK是实际下载的方法,待会介绍。但是写到后面发现DownloadManager已经帮我把这部分的权限做好了,在下载完apk之后会去申请权限。所以第一步相当于没用。。。

downloadAPK的实现

private void downloadAPK() {
    if (downloadId != 0) {
        //防止重复下载
        downloadManagerUtil.clearCurrentTask(downloadId);
    }
    downloadId = downloadManagerUtil.download(mDownloadUrl, "test.apk", "下载完成后,点击安装");
}

主要调用DownloadManager的工具类DownloadManagerUtil

public class DownloadManagerUtil {
    private Context mContext;

    public DownloadManagerUtil(Context context) {
        mContext = context;
    }

    public long download(String url, String title, String desc) {
        ToastUtil.toastShort("开始下载");
        Uri uri = Uri.parse(url);
        DownloadManager.Request req = new DownloadManager.Request(uri);
        //设置WIFI下进行更新,默认是所有网络都支持
        //        req.setAllowedNetworkTypes(DownloadManager.Request.NETWORK_WIFI);
        //下载中和下载完后都显示通知栏
        req.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED);
        //使用系统默认的下载路径 此处为应用内 /android/data/packages ,所以兼容7.0
        req.setDestinationInExternalPublicDir("test", "test.apk");
        //通知栏标题
        req.setTitle(title);
        //通知栏描述信息
        req.setDescription(desc);
        //设置类型为.apk
        req.setMimeType("application/vnd.android.package-archive");
        //获取下载任务ID
        DownloadManager dm = (DownloadManager) mContext.getSystemService(Context.DOWNLOAD_SERVICE);
        long requestId = dm.enqueue(req);
        //查询下载信息
        DownloadManager.Query query = new DownloadManager.Query();
        query.setFilterById(requestId);
        return requestId;
    }

    /**
     * 下载前先移除前一个任务,防止重复下载
     *
     * @param downloadId
     */
    public void clearCurrentTask(long downloadId) {
        DownloadManager dm = (DownloadManager) mContext.getSystemService(Context.DOWNLOAD_SERVICE);
        try {
            dm.remove(downloadId);
        } catch (IllegalArgumentException ex) {
            ex.printStackTrace();
        }
    }
}

主要使用的DownloadManager.Request去设置一些下载的属性,主要的注释都标识出来了,比较要注意的问题是这个路径req.setDestinationInExternalPublicDir("test", "test.apk");待会用FileProvider去获取路径的时候,要对应上这个路径,切记切记!要不然会提示软件安装包解析错误,因为路径没对上,FileProvider没找到对应的路径下的apk。所以就会提示解析错误,我在这个问题上弄了很久。

在使用DownloadManager下载完成之后,会发送一个广播,所以需要注册一个广播接收。

public class DownloadApkReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {

        if (intent.getAction().equals(DownloadManager.ACTION_DOWNLOAD_COMPLETE)) {
            //下载完成
            long downloadId = intent.getLongExtra(DownloadManager.EXTRA_DOWNLOAD_ID, -1L);
            // 根据获取到的ID,使用上面第3步的方法查询是否下载成功
            DownloadManager manager = (DownloadManager) context.getSystemService(Context.DOWNLOAD_SERVICE);
            DownloadManager.Query query = new DownloadManager.Query();
            query.setFilterById(downloadId);
            Cursor cursor = manager.query(query);
            if (!cursor.moveToFirst()) {
                cursor.close();
                return;
            }
            int status = cursor.getInt(cursor.getColumnIndex(DownloadManager.COLUMN_STATUS));
            String localFilename;
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                localFilename = cursor.getString(cursor.getColumnIndex(DownloadManager.COLUMN_LOCAL_URI));
            } else {
                localFilename = cursor.getString(cursor.getColumnIndex(DownloadManager.COLUMN_LOCAL_FILENAME));
            }
            if (status == DownloadManager.STATUS_SUCCESSFUL) {
                //                setPermission(localFilename);
                installApk(context, localFilename);
            }
        } else if (intent.getAction().equals(DownloadManager.ACTION_NOTIFICATION_CLICKED)) {
            //在下载过程中点击
            Intent viewDownloadIntent = new Intent(DownloadManager.ACTION_VIEW_DOWNLOADS);
            viewDownloadIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            context.startActivity(viewDownloadIntent);
        }

    }

    private void installApk(Context context, String fileName) {

        File file = new File(Environment.getExternalStorageDirectory().getAbsolutePath(), "test/test.apk");
        Intent install = new Intent(Intent.ACTION_VIEW);
        install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        if (Build.VERSION.SDK_INT >= 24) {
            //判读版本是否在7.0以上,7.0以上获取下载的apk文件要用FileProvider
            Uri apkUri = FileProvider.getUriForFile(context.getApplicationContext(),
                    "你的FileProvider的authority", file);
            install.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); //添加这一句表示对目标应用临时授权该Uri所代表的文件
            install.setDataAndType(apkUri, "application/vnd.android.package-archive");
        } else {
            install.setDataAndType(Uri.fromFile(file), "application/vnd.android.package-archive");
        }

        context.startActivity(install);
    }
}

广播的创建就不介绍了。不过要注意的是清单文件里面的intent-filter。要写成下面这样子才可以


    
        
        
    

DownloadManager.ACTION_DOWNLOAD_COMPLETE和DownloadManager.ACTION_NOTIFICATION_CLICKED这两个action分别对应不同的事件。
在installApk这个方法里面,涉及到7.0的FileProvider权限适配,FileProvider详细用法就不介绍了,百度一下一大堆,值得一提的是File file = new File(Environment.getExternalStorageDirectory().getAbsolutePath(), "test/test.apk");在第二个参数里面写的"test/test.apk"对应req.setDestinationInExternalPublicDir("test", "test.apk")这里的两个参数,一定要对应上,要不然找不到apk会报解析软件安装包错误安装失败的情况。

最后既然有广播,肯定要有注册的地方,

private void regist() {
    receiver = new DownloadApkReceiver();
    //register download success broadcast
    registerReceiver(receiver, new IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE));
}

最终效果图


效果图

效果还是挺好的,比自己写notification,service这些要简洁不少。

你可能感兴趣的:(DownloadManager使用(适配7.0,8.0))