分类:C#、Android、VS2015;

创建日期:2016-03-03

一、简介

除了上一节介绍的基本绑定以外,你还可以创建一个同时支持started和bound的服务。也就是说,服务可以通过调用 StartService() 来启动,这会使它一直保持运行,同时它也允许客户端通过调用BindService() 来与之绑定。

虽然你通常应该要实现 OnBind() 或 OnStartCommand() 中的一个,但有时需要同时实现两者。比如,音乐播放器的服务也许就需要同时实现后台运行和支持绑定。这样,activity就可以启动服务来播放音乐,并且音乐会一直播放下去,即使用户离开该应用程序也没关系,这个activity可以绑定播放服务来重新获得播放控制权。

另外,如果你的服务是started和bound的,那么服务启动后,系统将不会在所有客户端解除绑定时销毁它。取而代之的是,你必须通过调用StopSelf() 或 StopService() 显式终止此服务。

服务只在为绑定的应用程序组件工作时才会存活,因此,只要没有组件绑定到服务,系统就会自动销毁服务(你不需要像started服务中那样通过OnStartCommand()来终止一个bound服务)。

一旦OnBind方法返回绑定的实例,Android就会引发实现IServiceConnection接口的对象的ServiceConnected事件,然后,客户端就可以通过Binder引用它。

多个客户端可以同时连接到同一个服务上。不过,只有在第一个客户端绑定时,系统才会调用服务的 OnBind() 方法来获取 IBinder 。然后,系统会向后续请求绑定的客户端传送这同一个 IBinder ,而不再调用 OnBind() 。当最后一个客户端解除绑定后,系统会自动销毁服务(除非服务是同时通过 StartService() 启动的)。

二、示例2运行截图

该例子演示如何绑定到一个started服务。

单击【启动服务】按钮后,就会在左上角出现一个通知图标,下拉展开该图标,就可以看到来自服务的通知信息。

 

单击【调用服务中提供的方法】按钮,会显示调用服务中对应方法返回的结果,单击【停止服务】按钮,左上角的通知图标也会同时消失。

三、主要设计步骤

1、添加ch1702_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent">
<Button
android:id="@+id/ch1702_startService"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="启动服务" />
<Button
android:id="@+id/ch1702_callService"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="调用服务中提供的方法" />
<Button
android:id="@+id/ch1702_stopService"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="停止服务" />
</LinearLayout>

2、添加ch1702Service.cs

using Android.App;
using Android.Content;
using Android.OS;
using Android.Widget;
using System.Threading; namespace MyDemos.SrcDemos
{
[Service]
[IntentFilter(new string[] { action })]
public class ch1702Service : Service
{
public const string action = "ServiceDemo.ch1702Service";
ch1702ServiceBinder binder; public override StartCommandResult OnStartCommand(Intent intent, StartCommandFlags flags, int startId)
{
Notification notification = GetNotification();
//启动前台任务
StartForeground((int)NotificationFlags.ForegroundService, notification); var t = new Thread(() =>
{
//发送通知
var m = (NotificationManager)GetSystemService(NotificationService);
m.Notify((int)NotificationFlags.ForegroundService, notification); Thread.Sleep(); //延时50秒模拟长时间运行的服务 //停止前台任务
StopForeground(true);
//停止后台服务
StopSelf();
});
t.Start();
return StartCommandResult.NotSticky;
} private Notification GetNotification()
{
var pendingIntent = PendingIntent.GetActivity(this,
(int)NotificationFlags.ForegroundService,
new Intent(this, typeof(MainActivity)),
PendingIntentFlags.UpdateCurrent);
Notification.Builder builder = new Notification.Builder(this)
.SetContentTitle("来自ch1702Service的通知")
.SetContentText("正在前台运行ch1702Service")
.SetContentIntent(pendingIntent)
.SetSmallIcon(Resource.Drawable.Icon);
Notification notification = builder.Build();
return notification;
} public override IBinder OnBind(Android.Content.Intent intent)
{
binder = new ch1702ServiceBinder(this);
return binder;
} public string GetText()
{
return "这是来自ch1702Service的信息";
} public override void OnDestroy()
{
var myHandler = new Handler();
myHandler.Post(() =>
{
Toast.MakeText(this, "正在停止ch1702Service", ToastLength.Long).Show();
});
base.OnDestroy();
}
} public class ch1702ServiceBinder : Binder
{
ch1702Service service; public ch1702ServiceBinder(ch1702Service service)
{
this.service = service;
} public ch1702Service GetMyService()
{
return service;
}
}
}

3、添加ch1702MainActivity.cs

using Android.App;
using Android.Content;
using Android.OS;
using Android.Widget;
using Android.Content.Res; namespace MyDemos.SrcDemos
{
[Activity(Label = "ch1702MainActivity")]
public class ch1702MainActivity : Activity
{
bool isBound = false;
bool isConfigurationChange = false;
ch1702ServiceBinder binder;
ch1702ServiceConnection myServiceConnection;
Intent serviceIntent; protected override void OnCreate(Bundle savedInstanceState)
{
base.OnCreate(savedInstanceState);
SetContentView(Resource.Layout.ch1702_main); var start = FindViewById<Button>(Resource.Id.ch1702_startService);
start.Click += delegate
{
serviceIntent = new Intent(ch1702Service.action);
StartService(serviceIntent);
}; var callService = FindViewById<Button>(Resource.Id.ch1702_callService);
callService.Click += delegate
{
if (isBound)
{
RunOnUiThread(() =>
{
string text = binder.GetMyService().GetText();
Toast.MakeText(this, text, ToastLength.Short).Show();
});
}
}; var stop = FindViewById<Button>(Resource.Id.ch1702_stopService);
stop.Click += delegate
{
StopService(serviceIntent);
if (isBound)
{
UnbindService(myServiceConnection);
isBound = false;
}
}; // 如果配置改变了(比如旋转了屏幕),则恢复连接
if (myServiceConnection != null)
{
binder = myServiceConnection.Binder;
}
} protected override void OnStart()
{
base.OnStart();
var myServiceIntent = new Intent(ch1702Service.action);
myServiceConnection = new ch1702ServiceConnection(this);
BindService(myServiceIntent, myServiceConnection, Bind.AutoCreate);
} public override void OnConfigurationChanged(Configuration newConfig)
{
isConfigurationChange = true;
base.OnConfigurationChanged(newConfig);
} protected override void OnDestroy()
{
base.OnDestroy(); if (!isConfigurationChange)
{
if (isBound)
{
UnbindService(myServiceConnection);
isBound = false;
}
}
} private class ch1702ServiceConnection : Java.Lang.Object, IServiceConnection
{
private ch1702MainActivity activity;
public ch1702ServiceBinder Binder { get; private set; } public ch1702ServiceConnection(ch1702MainActivity activity)
{
this.activity = activity;
} public void OnServiceConnected(ComponentName name, IBinder service)
{
var myServiceBinder = service as ch1702ServiceBinder;
if (myServiceBinder != null)
{
activity.binder = myServiceBinder;
activity.isBound = true; // 即使配置发生了改变,仍保持绑定到该实例
Binder = myServiceBinder;
}
} public void OnServiceDisconnected(ComponentName name)
{
activity.isBound = false;
}
}
}
}

【Android】17.3 Activity与StartedService的绑定的更多相关文章

  1. 【Android】17.4 Activity与IntentService的绑定

    分类:C#.Android.VS2015: 创建日期:2016-03-03 一.简介 本示例通过AlarmManager类以固定的时间间隔调用服务(每隔2秒更新一次随机生成的股票数据).如果将此示例的 ...

  2. Android 组件系列-----Activity的传值和回传值

    在这篇随笔里将讲解Activity的传值和回传值. 一.通过startActivity来进行Activity的传值 在Android中,如果我们要通过一个Activity来启动另一个Activity, ...

  3. Android Service与Activity之间通信的几种方式

    在Android中,Activity主要负责前台页面的展示,Service主要负责需要长期运行的任务,所以在我们实际开发中,就会常常遇到Activity与Service之间的通信,我们一般在Activ ...

  4. 【原创】Android 从一个Activity跳转到另外一个Activity

    Android四大组件activity使用,实现两个activity之间的跳转 基本流程:创建两个activity-将其中一个activity中组件作为事件源-通过组件事件的处理借助intent对象实 ...

  5. Android中半透明Activity效果另法

    Android中的Activity有没有类似于像Windows程序样的窗口式显示呢? 答案当然是有. 下图就是一个窗口式Activity的效果图: 下面就说说实现过程: 首先看看AndroidMani ...

  6. [Android笔记1]Activity+Layout+Button

    线性布局(LinearLayout)是指view对象在父view中可按水平或垂直方向线性排列. 相对布局(RelativeLayout)是指view对象的排列依赖于各对象之间的相对位置. 下面是展示两 ...

  7. android 基础02 - Activity 的生命周期及状态

    返回栈 Android 中的 Activity 是可以层叠的,当我们启动一个新的 Activity 时,就会覆盖在原有的 Activity 之上, 点击 Back 会销毁当前 Activity,下面的 ...

  8. Android application 和 activity 标签详解

    extends:http://blog.csdn.net/self_study/article/details/54020909 Application 标签 android:allowTaskRep ...

  9. 【Android实验】第一个Android程序与Activity生命周期

    目录 第一个Android程序和Activity生命周期 实验目的 实验要求 实验过程 1. 程序正常启动与关闭 2. 外来电话接入的情况 3. 外来短信接入的情况 4. 程序运行中切换到其他程序(比 ...

随机推荐

  1. spring_restful_json_jdbc

    使用Spring MVC +JDBC 实现输出Json数据和视图两种形式 最后面有源代码 从web.xml開始配置: 声明定义两个Servlet各自是输出视图和json <!-- 声明一个Ser ...

  2. Appium Python 四:怎样获取APP的Package以及Activity

    看到一篇很好的博客:[Android测试][随笔]获得App的包名和启动页Activity 除了博客上的方法,我还找到两种方法: 方法一:aapt 前提需要使用SDK Manager.exe 下载 A ...

  3. PHP表单- PHP $_GET 变量

    PHP $_GET 变量 在 PHP 中,预定义的 $_GET 变量用于收集来自 method="get" 的表单中的值. $_GET 变量 预定义的 $_GET 变量用于收集来自 ...

  4. win7官方下载地址

    win7官方下载地址 http://microsoftstore.me/category/microsoft/windows/

  5. MVC Linq动态排序

    在nuget 中searh System.Linq.Dynamic 安装对应的版本, 这样都可以使用了 var orderExpression = string.Format("{0} {1 ...

  6. Spring Data MongoDB 三:基本文档查询(Query、BasicQuery)(一)

    一.简单介绍 Spring Data  MongoDB提供了org.springframework.data.mongodb.core.MongoTemplate对MongoDB的CRUD的操作,上一 ...

  7. const对象默觉得文件的局部变量

    const 定义的对象为一个常量不能被改动. 这个想必大家都知道. 这里仅仅是介绍const对象默觉得文件的局部变量         当一个非const变量在一个c或cpp文件里为全局时,它在整个程序 ...

  8. 简单四步開始树莓派上的Docker之旅

    大概这篇博文发表之后,应该算是我个人的第一篇翻译作品了,翻译的可能不是非常到位,望各位看官大刀砍过来. 原文链接:http://resin.io/blog/docker-on-raspberry-pi ...

  9. 别人的spring学习入门笔记

    http://elf8848.iteye.com/blog/875830  可以做入门参考,英语好可以阅读spring的spring-framework-reference 更多学习blog http ...

  10. glusterFS分布式存储部署流程

    转自:http://bangbangba.blog.51cto.com/3180873/1712061 GlusterFS是一款非常易于使用的分布式文件存储系统,实现了全部标准POSIX接口,并用fu ...