菜鸟的博客请多多指教

最近做了一个新功能,更新APK的功能

1、更新APK是一个耗时的任务,我采用了一个服务来做,上次在网上看到服务是在主线程里面,自己也测试了下,数据是真的

所以下载动作还必须在服务里面开一个线程来处理下载任务,在写下载任务的时候,由于经验不够,中间出了很多错,下载要分下载开始,下载中,下载异常,下载结束,

2、在后台下载,对用户来说,是看不见的,无法与用户交流,所以在开启服务的时候我就初始化了一个通知,通知在服务开启是初始化,所以其实通知也是在UI线程中,所以想在下载线程中更新通知进度栏,必须通过机制,来更新通知栏的进度条,但是后来组长又给你我一个需求,在点击通知栏时,在更新界面弹出一个对话框,显示是否要停止更新,这里就涉及到一个新的问题,怎样在Service里面获取Activity呢,找了很多资料,最后找到了一个ActivityManager系统服务,他可以得到指定包的任务栈,得到栈顶的Activity,这样就可以在Service中获取Activity里面了

            String MY_PKG_NAME = "com.example.sunny.text333";
int i=0;
ActivityManager manager = (ActivityManager) MyApplication.getInstance().getSystemService(Context.ACTIVITY_SERVICE);
List<ActivityManager.RunningTaskInfo> list = manager.getRunningTasks(100);
for (ActivityManager.RunningTaskInfo info: list){
if(info.topActivity.getPackageName().equals(MY_PKG_NAME) || info.baseActivity.getPackageName().equals(MY_PKG_NAME)){
break;
}
i++;
}
ActivityManager.RunningTaskInfo info=list.get(i);
String className = info.topActivity.getClassName();//得到栈顶的Activity
Intent intent = null;
try {
intent = new Intent(this,Class.forName(className));
intent.addCategory(Intent.CATEGORY_LAUNCHER);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK| Intent.FLAG_ACTIVITY_SINGLE_TOP);
Bundle bundle = new Bundle();
bundle.putString("flag","取消");
intent.putExtras(bundle);
} catch (ClassNotFoundException e) {
e.printStackTrace();
Log.e("没有找到Activity",""+className);
}
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP
|Intent.FLAG_ACTIVITY_NEW_TASK);
pendingIntent = PendingIntent.getActivity(this, 0,intent,PendingIntent.FLAG_UPDATE_CURRENT);
notification.contentIntent =pendingIntent;
Log.e("通知的Activity名称",""+className);

下载线程:

class DownloadThread extends Thread {
String url;
MyHandler handler; public DownloadThread(String url, MyHandler handler) {
this.url = url;
this.handler = handler;
} @Override
public void run() { FileOutputStream fos = null;
InputStream io = null;
try {
handler.sendEmptyMessage(0);
URL u = new URL(url);
Log.e("URL","____________"+u);
HttpURLConnection connection = (HttpURLConnection) u.openConnection();
connection.setConnectTimeout(30000);
connection.setReadTimeout(30000);
io = connection.getInputStream(); if (io != null) {
File file = new File(Environment.getExternalStorageDirectory() + "/" + "mapp.apk");
byte[] buff = new byte[1024];
if (file.exists()) {
file.delete();
} else {
file.createNewFile();
}
fos = new FileOutputStream(file);
int readNum = 0;
int lastPress = 0;//上一次的进度值
int press = 0;//
int fileLength = 0;
int count = 0;//记录文件的读取个数
while (isrunning) {
readNum = io.read(buff);
fileLength = connection.getContentLength();
count += readNum;
press = (int) ((float) count / (float) (fileLength - 1) * 100.0);//下载总大小除以总的大小*100
BigDecimal bd = new BigDecimal(count);
BigDecimal bd1 = new BigDecimal(fileLength - 1);
if (readNum == -1 && bd.toPlainString().equals(bd1.toPlainString())) {// 下载完成
Log.e("sdas", "下载完成");
Message message = new Message();
message.obj = file;
message.what = 2;
handler.sendMessage(message);
break;
} else if (readNum == -1 && !bd.toPlainString().equals(bd1.toPlainString())) {
Log.e("sdas", "下载失败");
handler.sendEmptyMessage(4);
break;
} else if (readNum > 0) {
fos.write(buff, 0, readNum);
// Log.e("sdas", "readNum:" + readNum);
if (press - lastPress >= 1) {
lastPress = press;
// Log.e("sdas", "press:" + press);
Message message = new Message();
message.what = 1;
message.obj = press + "";
handler.sendMessage(message);
}
}
}
fos.flush();
fos.close();
io.close();
}
} catch (MalformedURLException e) {
handler.sendEmptyMessage(4);
e.printStackTrace();
} catch (IOException e) {
handler.sendEmptyMessage(4);
e.printStackTrace();
} finally {
try {
if (fos != null) {
fos.flush();
fos.close();
}
if (io != null) {
io.close();
}
} catch (IOException e) { } }
}
}

MyHandler的代码:

class MyHandler extends Handler {
@Override
public void handleMessage(Message msg) { super.handleMessage(msg);
if (msg != null) {
switch (msg.what) {
case 0:// 初始化通知栏和进度条
views.setTextViewText(R.id.tvProcess, "开始下载……");
views.setTextColor(R.id.tvProcess, Color.BLACK);
views.setProgressBar(R.id.app_down_progressBar, 100, 0, false);
notification.defaults = Notification.DEFAULT_LIGHTS;
notification.contentView = views;
nm.notify(NOTIFICATIO_UPDATE_SOFT, notification);
MyApplication.getInstance().setIsupdateing(true);
break;
case 1:
int download_precent = Integer.parseInt(msg.obj.toString());// 获取下载进度
// 更新状态栏上的下载进度信息
views.setTextViewText(R.id.tvProcess, "已下载" + download_precent + "%");
views.setTextColor(R.id.tvProcess, Color.BLACK);
views.setProgressBar(R.id.app_down_progressBar, 100, download_precent, false);
notification.defaults = Notification.DEFAULT_LIGHTS;
notification.contentView = views;
nm.notify(NOTIFICATIO_UPDATE_SOFT, notification);
MyApplication.getInstance().setIsupdateing(true); break;
case 2:
MyApplication.getInstance().setIsupdateing(false);
/* if( unIntallapk()){
Log.e("文件卸载",""+unIntallapk()); }*/
Instanll((File) msg.obj); nm.cancel(NOTIFICATIO_UPDATE_SOFT);
// 停止掉当前的服务
stopSelf();
break;
case 3:// 停止下载
MyApplication.getInstance().setIsupdateing(false);
// 停止掉当前的服务
nm.cancel(NOTIFICATIO_UPDATE_SOFT);
stopSelf();
break;
case 4:// 下载异常
MyApplication.getInstance().setIsupdateing(false);
views.setTextViewText(R.id.tvProcess, "" + "已停止下载");
views.setTextColor(R.id.tvProcess, Color.BLUE);
// 在任务栏不被用户手动删除
notification.flags = Notification.FLAG_AUTO_CANCEL;
// 调用系统默认的声音提示以及震动
notification.defaults = Notification.DEFAULT_ALL;
notification.contentView = views;
nm.cancel(NOTIFICATIO_UPDATE_SOFT);
nm.notify(NOTIFICATIO_UPDATE_SOFT, notification);
break;
default:
break;
}
}
}
}

android APK更新的更多相关文章

  1. Android Apk增量更新

    前言 有关APK更新的技术比较多,例如:增量更新.插件式开发.热修复.RN.静默安装. 下面简单介绍一下: 什么是增量更新?   增量更新就是原有app的基础上只更新发生变化的地方,其余保持原样. 与 ...

  2. 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新

    [原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...

  3. 【原】Android热更新开源项目Tinker源码解析系列之二:资源文件热更新

    上一篇文章介绍了Dex文件的热更新流程,本文将会分析Tinker中对资源文件的热更新流程. 同Dex,资源文件的热更新同样包括三个部分:资源补丁生成,资源补丁合成及资源补丁加载. 本系列将从以下三个方 ...

  4. Android 数据库管理— — —更新数据

    <?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android=" ...

  5. Android 增量更新(BSDiff / bspatch)

    Android 增量更新 BSDiff / bspatchhttp://www.daemonology.net/bsdiff/android的代码目录下 \external\bsdiff bsdiff ...

  6. android apk 自我保护技术-加密apk

    经过了忙碌的一周终于有时间静下来写点东西了,我们继续介绍android apk防止反编译技术的另一种方法.前两篇我们讲了加壳技术(http://my.oschina.net/u/2323218/blo ...

  7. 如何加密android apk

    经过了忙碌的一周终于有时间静下来写点东西了,我们继续介绍android apk防止反编译技术的另一种方法.前两篇我们讲了加壳技术(http://my.oschina.net/u/2323218/blo ...

  8. android开发 更新升级安装到一半自动闪退

    如题:android开发 更新升级安装到一半自动闪退,,,解决办法,如下(红色为我新增的代码) /**     * 安装APK文件     */    private void installApk( ...

  9. Android热更新开源项目Tinker集成实践总结

    前言 最近项目集成了Tinker,开始认为集成会比较简单,但是在实际操作的过程中还是遇到了一些问题,本文就会介绍在集成过程大家基本会遇到的主要问题. 考虑一:后台的选取 目前后台功能可以通过三种方式实 ...

随机推荐

  1. sqlserver数据库 Schema

    //读取数据库中所有的数据库 USE MASTER DECLARE @is_policy_automation_enabled bit SET @is_policy_automation_enable ...

  2. ES6里箭头函数的陷阱

    ECMAScript 6新增了箭头函数 原来的匿名函数 function(){},现在可以简化成()=>{} 看起来高大上,像C#什么的语法. 但是箭头函数的this对象,不能更改,总是指向函数 ...

  3. 指令的Link函数和Scope

    指令生成出的模板其实没有太多意义,除非它在特定的scope下编译.默认情况下,指令并不会创建新的子scope.更多的,它使用父scope.也就是说,如果指令存在于一个controller下,它就会使用 ...

  4. Windows Azure Storage 之 Retry Policy (用来处理短暂性错误-Transient Fault)

    在使用Windows Azure Storage Service 的时候, 通常会遇到各种各样的问题. 例如网络连接不稳定,导致请求没有发出去.删除一个Blob Container 之后又立刻创建同名 ...

  5. CoreData 轻量级迁移

    CoreData 轻量级迁移 Core Data 的轻量级迁移可以处理对数据模型进行的简单更改,例如将新属性添加到现有实体中,轻量级迁移基本上与普通迁移相同,区别就是不需要映射模型,因为Core Da ...

  6. redhat6.3下源码编译方式安装最新版git

    在linux下安装git,通过yum方式安装的不是最新版本. 要替换最新版需要以下操作. 1.安装依赖包 # yum install curl-devel expat-devel gettext-de ...

  7. python 编辑文件时路径问题解决方法:文件或者目录不存在、文件编辑后无法保存等(以编辑xml文件为例)

    1.获取工程所在根路径:根路径=os.path.dirname(os.path.abspath('__file__')) 2.将获取的根路径和相对路径组合:组合路径=os.path.join(根路径, ...

  8. thinkPHP5.x 更新字段为 NULL

    简介 tp5.x 提供了丰富的数据模型和数据库操作的方法,只要涉及 think\Model think\Query等,其中有一个软删除的 feature,可以指定字段$deleteTime来标记 re ...

  9. [已解决][HTML5]基于WebSocket开发聊天室应用

    WebSocket示例java的比较少,大部分是nodejs的,比较有名的是socket.io的chat, 借用下他的前端实现一套java的,后端基于https://github.com/genera ...

  10. 获取WOED和EXCEL的公用方法

    1. 需要传入word地址 /// <summary> /// 获取WORD内容 /// </summary> /// <param name="docFile ...