欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android中 service组件详解

程序员文章站 2024-03-06 11:06:13
 service组件跟activity组件及其类似,可以说service是没有界面的activity, 当然service的生命周期和activity还是有...

 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 帮它实现了。