分享好友 移动开发首页 频道列表

从 APP 升级更新浅谈安卓 Service 知识

Android开发  2016-12-01 11:140
博主原创转载请注明作者MskyAndroid
出处http://blog.csdn.net/AndroidMsky/article/details/53407925

GITHUB链接:

https://github.com/AndroidMsky/UpdateApp

本文将简述一个APP升级功能,通过前台service实时下载一个APK然后跳入安装界面.

 从 APP 升级更新浅谈安卓 Service 知识

 从 APP 升级更新浅谈安卓 Service 知识

首先回顾一下Service的知识点,service可翻译为服务,你可以暂时把它理解为一个没有界面的Activity,但是他的生命周期又和activity有所不同。但又大致相同,我认为唯一需要特别注意的就是onbind方法,是绑定的意思。也就是说一个service自己在黑暗中运行,要如何给他指指路呢,那么就通过onbind方法,绑定这个service。说起绑定这个词肯定很多人又开始晕了,其实可以这样理解,一个Activity去通过ServiceConnection绑定一个service,就是从Activity中拿取一个service中的对象,这个对象往往有一些可以控制service做什么的方法(这并不是一定的,你可以写一些没什么用的方法因为这个类的方法都是自己定义的)。

好下面举个栗子:

本文通过更新按钮启动一个service去通过HttpURLConnection去下载一个网络的apk,并且可以通过获取进度按钮获取实时进度。实时获取进度,我们就用activity去绑定一个service来实现。

首先也就是我们刚才说的,activity拿取service的一个对象,这个对象的类型必须是Binder的子类:

class MyBinder extends Binder {

        public int getDownload() {
            Log.d("TAG", "getDownload ");
            return updateCount;

        }

    }

在service中声明该对象

private MyBinder mBinder = new MyBinder();

在service这个唯一的抽象方法中返回这个mBinder对象

@Override
    public IBinder onBind(Intent arg0) {
        return mBinder;
    }

service的工作就做完了,接下来就是activity中写一个实现serviceConnection接口的类:

class MyServiceConnection implements ServiceConnection {
        @Override
        public void onServiceConnected(ComponentName arg0, IBinder service) {
            Log.d("info", "Service Connection Success");

            myBinder = (UpdateService.MyBinder) service;

        }

        @Override
        public void onServiceDisconnected(ComponentName arg0) {
// TODO Auto-generated method stub
            Log.d("info", "Service Connection Filed");
//连接失败执行
        }

    }

可见两个方法,一个连接成功一个连接失败,连接成功方法中的IBinder service也就是传递给我们的那个连接二者的对象,操作这个对象的方法,去指挥service中的动作就可以了。

最后启动service并且绑定:

UpdateService.into(this, getString(R.string.app_name), url);
Intent intent = new Intent(this, UpdateService.class);
        bindService(intent,myServiceConnection , Context.BIND_AUTO_CREATE);

相信绑定这块大家都已经清楚了,接下来就看一下service的生命周期,我们在几个关键的生命周期打一下log。当我们点击更新按钮,创造并绑定service:

 从 APP 升级更新浅谈安卓 Service 知识

当我们每次点击获取进度按钮,也就是通过绑定得到那个对象中的方法,都会调用service中的方法执行。

特别需要注意的是即使调用

stopSelf();

service并没有完全结束,因为它还被绑定,必须解绑stop两者同时满足service才会停下来,我们就设定如果获取进度打到100的时候就进行解绑。

public void getprogress(View v){

        textView.setText("进度:"+myBinder.getDownload());
        if (myBinder.getDownload()==100)
            unbindService(myServiceConnection);


    }

 从 APP 升级更新浅谈安卓 Service 知识

接下来说一下前台service一个service如何穿上前台service的外衣就是在通知栏去创造一个通知它就升级为前台service,这样就有了更高的优先级不容易被系统杀死了,当然这个通知不是通过NotificationManager来显示出来的,是通过startForeground(1, notification);方法来显示出来,这样service就编程前台service了。

public void createNotification() {
        builder = new Notification.Builder(this);
        builder.setSmallIcon(R.mipmap.ic_launcher).setOngoing(false)
                .setContentText("正在下载")
                .setContentTitle("XXXX下载");
        builder.setAutoCancel(false);
        builder.setOngoing(true);
        builder.setTicker("XXXX正在下载");
        builder.setLargeIcon(BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher));
        builder.setProgress(100, 0, false);

        Notification notification = null;
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
            builder.setPriority(Notification.PRIORITY_MAX);
            notification = builder.build();
        } else {
            notification = builder.getNotification();
        }
        notification.icon = R.mipmap.ic_launcher;
        startForeground(1, notification);

    }

另外不要以为service在后台没有界面它就有自己的线程了,错错错,普通的service的代码默认还是跑在主线程中的,所以我们开启下载任务也是要重新开启线程来下载的。

private class DownLoadThread extends Thread {
        @Override
        public void run() {
            // TODO Auto-generated method stub
            Message message = new Message();
            try {
                long downloadSize = downloadUpdateFile(down_url,
                        FileUtil.updateFile.toString());
                if (downloadSize > 0) {
                    message.what = DOWN_OK;
                    handler.sendMessage(message);
                }
            } catch (Exception e) {
                message.what = DOWN_ERROR;
                handler.sendMessage(message);
            }
        }
    }

相信通过本文的浅谈,大家都可以灵活的去使用service了。

欢迎关注作者。欢迎评论讨论。欢迎拍砖。 如果觉得这篇文章对你有帮助,欢迎打赏, 欢迎star,Fork我的github。 喜欢作者的也可以Follow。也算对作者的一种支持。

本文Github代码链接

https://github.com/AndroidMsky/GlideScrollRecyclerView 欢迎加作者自营安卓开发交流群:308372687

 从 APP 升级更新浅谈安卓 Service 知识

—————————————————————————————

作者推荐:

安卓自定义view滚动数据显示

http://blog.csdn.net/androidmsky/article/details/53009886 RecyclerView

下拉刷新分页加载性能优化和Gilde配合加载三部曲

http://blog.csdn.net/androidmsky/article/details/53115818

打造企业级网络请求框架集合retrofit+gson+mvp

http://blog.csdn.net/androidmsky/article/details/52882722

安卓手机自动接起QQ视频秒变摄像头

http://blog.csdn.net/androidmsky/article/details/53066441 —————————————————————————————

查看更多关于【Android开发】的文章

展开全文
相关推荐
反对 0
举报 0
评论 0
图文资讯
热门推荐
优选好物
更多热点专题
更多推荐文章
Supporting Multiple Screens
术语和概念Screen size 屏幕尺寸又称「屏幕大小」,是屏幕对角线的物理尺寸。单位英寸 inch,比如 Samsung Note4 是 5.7 英寸。Resolution 屏幕分辨率屏幕纵横方向上物理像素的总数,比如 Samsung Note4 是 2560x1440,表示纵向有 2560 个像素,横向有 1440

0评论2017-02-05363

Android插件化(4):OpenAtlasの插件的卸载与更新
如果看过我的前两篇博客Android插件化(2):OpenAtlas插件安装过程分析和Android插件化(3):OpenAtlas的插件重建以及使用时安装,就知道在插件的安装过程中OpenAtlas做了哪些事,那么插件的卸载就只需要把持久化和内存中的内容移除即可。1.插件的卸载插件卸载的

0评论2017-02-05229

个人简历
吴朝晖/男/1993.1本科/南京师范大学中北学院信息系工作年限:1年以内技术博客:wuzhaohui026.github.ioGitHub:https://github.com/wuzhaohui026期望职位:Android开发(初级Android工程师)期望薪资:税前月薪5.5k~7k期望城市:常州工作经历常州慧展信息科技有

0评论2017-02-05126

Android插件化(五):OpenAtlasの四大组件的Hack
引言到目前为止,我们已经分析了OpenAtlas中插件的安装,卸载,更新,以及安装好插件之后组件类的加载过程,但是对于这些是如何引发的还不知道,比如,在宿主的一个Activit中调用startActivity()跳转到插件中的一个Activity,如何判断这个Activity在的插件是否

0评论2017-02-0598

更多推荐