热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android中service组件详解

Service是Android的四大组件之一,以下是我结合AndroidDoc和网上资料的学习总结,有不准确的地方请高手指出,互相学习嘛。。。

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


推荐阅读
  • Android 九宫格布局详解及实现:人人网应用示例
    本文深入探讨了人人网Android应用中独特的九宫格布局设计,解析其背后的GridView实现原理,并提供详细的代码示例。这种布局方式不仅美观大方,而且在现代Android应用中较为少见,值得开发者借鉴。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • 本文介绍如何在 Android 中通过代码模拟用户的点击和滑动操作,包括参数说明、事件生成及处理逻辑。详细解析了视图(View)对象、坐标偏移量以及不同类型的滑动方式。 ... [详细]
  • 本文详细介绍了如何使用PHP检测AJAX请求,通过分析预定义服务器变量来判断请求是否来自XMLHttpRequest。此方法简单实用,适用于各种Web开发场景。 ... [详细]
  • 本文探讨了 RESTful API 和传统接口之间的关键差异,解释了为什么 RESTful API 在设计和实现上具有独特的优势。 ... [详细]
  • 本文详细介绍了如何使用Spring Boot进行高效开发,涵盖了配置、实例化容器以及核心注解的使用方法。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • Python自动化处理:从Word文档提取内容并生成带水印的PDF
    本文介绍如何利用Python实现从特定网站下载Word文档,去除水印并添加自定义水印,最终将文档转换为PDF格式。该方法适用于批量处理和自动化需求。 ... [详细]
  • 在当前众多持久层框架中,MyBatis(前身为iBatis)凭借其轻量级、易用性和对SQL的直接支持,成为许多开发者的首选。本文将详细探讨MyBatis的核心概念、设计理念及其优势。 ... [详细]
  • 将Web服务部署到Tomcat
    本文介绍了如何在JDeveloper 12c中创建一个Java项目,并将其打包为Web服务,然后部署到Tomcat服务器。内容涵盖从项目创建、编写Web服务代码、配置相关XML文件到最终的本地部署和验证。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • 本文介绍如何在 Unity 的 XML 配置文件中,将参数传递给自定义生命周期管理器的构造函数。我们将详细探讨 CustomLifetimeManager 类的实现及其配置方法。 ... [详细]
  • 本文详细介绍了 Java 中 org.apache.xmlbeans.SchemaType 类的 getBaseEnumType() 方法,提供了多个代码示例,并解释了其在不同场景下的使用方法。 ... [详细]
author-avatar
一二三
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有