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。要写成下面这样子才可以

<receiver
    android:name=".mine.updateapp.DownloadApkReceiver"
    android:enabled="false"
    android:exported="false">
    <intent-filter>
        <action android:name="android.intent.action.DOWNLOAD_COMPLETE" />
        <action android:name="android.intent.action.DOWNLOAD_NOTIFICATION_CLICKED" />
    </intent-filter>
</receiver>

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这些要简洁不少。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,591评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,448评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,823评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,204评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,228评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,190评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,078评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,923评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,334评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,550评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,727评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,428评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,022评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,672评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,826评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,734评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,619评论 2 354

推荐阅读更多精彩内容