Android四大组件 之 service简单理解

443 查看

    service组件跟activity组件及其类似,可以说service是没有界面的activity,
当然service的生命周期和activity还是有一定的差别的。
    service组件一般用在什么地方的,上面讲了service组件没有界面,不用跟用户直接交互,
所以service组件一般运行在后台。比如做一些不需要界面的数据处理等等。

开发service需要两个步骤:
    1,定义一个基础service的子类。
    2,在AndroidManifest.xml 文件中配置该service。

怎么启动service呢,想想启动activity是不是有两种方法:
    startActivity(intent),
    startActivityForResult(intent)
那么启动service也有两种方法:
    startService(intent),
    bindService(Intent service,ServiceConnection conn,int flags),
两者有什么区别可以先看下面的代码:

public class BindService extends Service
{
    private int count;
    private boolean quit;
    // 定义onBinder方法所返回的对象
    private MyBinder binder = new MyBinder();
    // 通过继承Binder来实现IBinder类
    public class MyBinder extends Binder
    {
        public int getCount()
        {
            // 获取Service的运行状态:count
            return count;
        }
    }
    // 必须实现的方法
    @Override
    public IBinder onBind(Intent intent)
    {
        System.out.println("Service is Binded");
        // 返回IBinder对象
        return binder;
    }
    // Service被创建时回调该方法。
    @Override
    public void onCreate()
    {
        super.onCreate();
        System.out.println("Service is Created");
        // 启动一条线程、动态地修改count状态值
        new Thread()
        {
            @Override
            public void run()
            {
                while (!quit)
                {
                    try
                    {
                        Thread.sleep(1000);
                    }
                    catch (InterruptedException e)
                    {
                    }
                    count++;
                }
            }
        }.start();        
    }
    // Service被断开连接时回调该方法
    @Override
    public boolean onUnbind(Intent intent)
    {
        System.out.println("Service is Unbinded");
        return true;
    }
    // Service被关闭之前回调。
    @Override
    public void onDestroy()
    {
        super.onDestroy();
        this.quit = true;
        System.out.println("Service is Destroyed");
    }
    
    @Override
    public void onRebind(Intent intent) 
    {
        super.onRebind(intent);
        this.quit = true;
        System.out.println("Service is ReBinded");
    }    
}
上面的Service的作用是 简单的开启一个线程,每 1秒钟 count++,这个count数据
通过 binder对象 传递给 访问者。

待会再做详解,先看下面的代码怎么启动Service,并得到 Service的 count数据
public class MainActivity extends Activity
{
    Button startService_bnt , bindService_bnt;
    // 保持所启动的Service的IBinder对象
    BindService.MyBinder binder;
 
    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
 
        startService_bnt = (Button) findViewById(R.id.start_bnt);
        bindService_bnt = (Button) findViewById(R.id.bind_bnt);
 
        //创建启动Service的Intent
         Intent intent = new Intent(this,BindService.class);
 
         startService_bnt.setOnClickListener(new OnClickListener()
        {
            @Override
            public void onClick(View source)
            {
                //绑定指定Serivce
                startService(intent);
            }
        });    
        
        bindService_bnt.setOnClickListener(new OnClickListener()
        {
            @Override
            public void onClick(View source)
            {
                //绑定指定Serivce
                bindService(intent , conn , Service.BIND_AUTO_CREATE);    
                
                Toast.makeText(MainActivity.this
                    , "Serivce的count值为:" + binder.getCount()
                    , 4000)
                    .show();
            }
        });
 
 
    }
    
        // 定义一个ServiceConnection对象
    private ServiceConnection conn = new ServiceConnection()
    {
        // 当该Activity与Service连接成功时回调该方法
        @Override
        public void onServiceConnected(ComponentName name
            , IBinder service)
        {
            System.out.println("--Service Connected--");
            // 获取Service的onBind方法所返回的MyBinder对象
            binder = (BindService.MyBinder) service;
        }
        // 当该Activity与Service断开连接时回调该方法
        @Override
        public void onServiceDisconnected(ComponentName name)
        {
            System.out.println("--Service Disconnected--");            
        }
    };
}

上面activity定义了两个按钮,点击两个按钮有两种不同的方法启动Service:

 startService(intent),
 bindService(Intent service,ServiceConnection conn,int flags),
 

现在来讲解一下两种启动方式的区别,并解释上面的代码。

startService(intent)启动Service呢它不具有与访问者交互的能力,就像
activity 的 startActivity(),它不能从新启动的activity拿到返回数据一样 

而bindService(Intent service,ServiceConnection conn,int flags),就不一样了
访问者能从启动的Service 拿到数据,怎么拿到的呢,bindService的第二个参数 conn,
该参数是一个 ServiceConnection  对象,当访问者与Service连接成功 就会回调
ServiceConnection  的 onServiceConnected() 方法 ,上面的程序就是在这个回调方法
里面拿到 IBinder  对象的。

可以在看一下

// 定义一个ServiceConnection对象
    private ServiceConnection conn = new ServiceConnection()
    {
        // 当该Activity与Service连接成功时回调该方法
        @Override
        public void onServiceConnected(ComponentName name
            , IBinder service)
        {
            System.out.println("--Service Connected--");
            // 获取Service的onBind方法所返回的MyBinder对象
            binder = (BindService.MyBinder) service;
        }
        // 当该Activity与Service断开连接时回调该方法
        @Override
        public void onServiceDisconnected(ComponentName name)
        {
            System.out.println("--Service Disconnected--");            
        }
    };
简单点也就是说 访问者通过 bindService 绑定到 Service,绑定成功后会回调
ServiceConnection 中的 onServiceConnected()方法,这个方法里面有
IBinder service 参数,这个参数就是 Service暴露给 访问者的对象,访问者拿到这个对象
就可以访问 Service的数据了
这就是 访问者与Service数据交互的原理,是通过 IBinder 对象来传递的。

可能到这这里你还对 binder = (BindService.MyBinder) service;这句代码不理解。

 你肯能觉得 拿到的IBinder 对象不应该是上面Service代码中
onBind 方法返回的 binder 才是嘛,怎么 强转成 BindService.MyBinder 对象了。
而且返回的 binder  也没 count数据,访问者怎么就能 binder.getCount() 得到数据呢。
@Override
    public IBinder onBind(Intent intent)
    {
        System.out.println("Service is Binded");
        // 返回IBinder对象
        return binder;
    }

别忘了 上面Service代码里面还对 IBinder 对象进行处理

// 通过继承Binder来实现IBinder类
    public class MyBinder extends Binder
    {
        public int getCount()
        {
            // 获取Service的运行状态:count
            return count;
        }
    }

Binder 是 IBinder 的 实现类,MyBinder 继承Binder 并在里面定义了个方法。

那么 拿到  IBinder  对象 就相当于 拿到  MyBinder 对象,就可以访问 getCount方法了
这也是 为什么 binder = (BindService.MyBinder) service; 进行强转,并且
binder.getCount() 可以拿到 count 数据,因为 IBinder 里面并没有业务实现,是MyBinder 
帮它实现了。

下一篇讲讲利用 AIDL 技术 实现跨进程通信 ,如果上面这篇感觉我讲的还看的算清晰的话
可以私信我尽快更新下一篇哦