Android学习笔记(九)一个例子弄清Service与Activity通信
上一篇博文主要整理了Service的创建、绑定过程,本篇主要整理一下Service与Activity的通信方式。包括在启动一个Service时向它传递数据、怎样改变运行中的Service中得数据和侦听Service内数据的改变。
本篇将写一个demo来说明以下三个问题:
1、怎样在启动一个Service时向它传递数据
关键点:Intent传值,onStartCommand()接收。
2、怎样向运行的Service中同步数据
关键点:通过onBind()获取Service实例,然后再调用Binder中的相关方法。
3、怎样侦听Service中数据变化
关键点:通过回调函数达到目的。
一、准备Service
先贴出Service的详细代码,然后再慢慢分析
public class MyService extends Service {
private String data = "默认消息";
private boolean serviceRunning = false; // 必须实现的方法,用于返回Binder对象
@Override
public IBinder onBind(Intent intent) {
System.out.println("--onBind()--");
return new MyBinder();
} public class MyBinder extends Binder {
MyService getService() {
return MyService.this;
} public void setData(String data) {
MyService.this.data = data;
}
} // 创建Service时调用该方法,只调用一次
@Override
public void onCreate() {
super.onCreate();
System.out.println("--onCreate()--");
serviceRunning = true;
new Thread() {
@Override
public void run() {
int n = 0;
while (serviceRunning) {
n++;
String str = n + data;
System.out.println(str);
if (dataCallback != null) {
dataCallback.dataChanged(str);
}
try {
sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
}.start();
} // 每次启动Servcie时都会调用该方法
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
System.out.println("--onStartCommand()--");
data = intent.getStringExtra("data");
return super.onStartCommand(intent, flags, startId);
} // 解绑Servcie调用该方法
@Override
public boolean onUnbind(Intent intent) {
System.out.println("--onUnbind()--");
return super.onUnbind(intent);
} // 退出或者销毁时调用该方法
@Override
public void onDestroy() {
serviceRunning = false;
System.out.println("--onDestroy()--");
super.onDestroy();
} DataCallback dataCallback = null; public DataCallback getDataCallback() {
return dataCallback;
} public void setDataCallback(DataCallback dataCallback) {
this.dataCallback = dataCallback;
} // 通过回调机制,将Service内部的变化传递到外部
public interface DataCallback {
void dataChanged(String str);
} }
代码分析:我们都知道,通过startService启动一个Service时,Service会调用生命周期函数onStartCommand(),在代码中创建一个Service,在onStartCommand()方法中获取从Activity传递过来的数据,并在Service的onCreate()方法中开启一个新的线程,使其循环调用回调函数,以达到通知外界信息改变的目的。并在Service中通过Binder类,将Service与Activity链接起来,以实现信息同步。
二、准备布局文件
布局文件比较简单,直接贴出,就不分析了,activity_main.xml如下:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"> <TextView
android:id="@+id/tv_out"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="显示区域" /> <EditText
android:id="@+id/et_data"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:ems="10" > <requestFocus />
</EditText> <Button
android:id="@+id/btn_start_service"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="startService" /> <Button
android:id="@+id/btn_stop_service"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="stopService" /> <Button
android:id="@+id/btn_bind_service"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="bindService" /> <Button
android:id="@+id/btn_unbind_service"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="unbindService" /> <Button
android:id="@+id/btn_sync_data"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="同步数据" /> </LinearLayout>
三、准备Activity
MainActivity代码如下:
public class MainActivity extends Activity implements OnClickListener { private Intent intent = null;
private Button btn_start_service;
private Button btn_stop_service;
private Button btn_bind_service;
private Button btn_unbind_service;
private Button btn_sync_data;
private EditText et_data;
private TextView tv_out;
MyServiceConn myServiceConn;
MyService.MyBinder binder = null; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
intent = new Intent(this, MyService.class);
myServiceConn = new MyServiceConn();
setOnClick();
} @Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.btn_start_service:
//用intent启动Service并传值
intent.putExtra("data", et_data.getText().toString());
startService(intent);
break;
case R.id.btn_stop_service:
//停止Service
stopService(intent);
break;
case R.id.btn_bind_service:
//绑定Service
bindService(intent, myServiceConn, Context.BIND_AUTO_CREATE);
break;
case R.id.btn_unbind_service:
//解绑Service
if (binder != null) {
unbindService(myServiceConn);
}
break;
case R.id.btn_sync_data:
//注意:需要先绑定,才能同步数据
if (binder != null) {
binder.setData(et_data.getText().toString());
}
break;
default:
break;
}
} class MyServiceConn implements ServiceConnection {
// 服务被绑定成功之后执行
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// IBinder service为onBind方法返回的Service实例
binder = (MyService.MyBinder) service;
binder.getService().setDataCallback(new MyService.DataCallback() {
//执行回调函数
@Override
public void dataChanged(String str) {
Message msg = new Message();
Bundle bundle = new Bundle();
bundle.putString("str", str);
msg.setData(bundle);
//发送通知
handler.sendMessage(msg);
}
});
} @SuppressLint("HandlerLeak")
Handler handler = new Handler() {
public void handleMessage(android.os.Message msg) {
//在handler中更新UI
tv_out.setText(msg.getData().getString("str"));
};
}; // 服务奔溃或者被杀掉执行
@Override
public void onServiceDisconnected(ComponentName name) {
binder = null;
}
} private void loadUI() {
btn_start_service = (Button) findViewById(R.id.btn_start_service);
btn_stop_service = (Button) findViewById(R.id.btn_stop_service);
btn_bind_service = (Button) findViewById(R.id.btn_bind_service);
btn_unbind_service = (Button) findViewById(R.id.btn_unbind_service);
btn_sync_data = (Button) findViewById(R.id.btn_sync_data);
et_data = (EditText) findViewById(R.id.et_data);
tv_out = (TextView) findViewById(R.id.tv_out);
} private void setOnClick() {
loadUI();
btn_start_service.setOnClickListener(this);
btn_stop_service.setOnClickListener(this);
btn_bind_service.setOnClickListener(this);
btn_unbind_service.setOnClickListener(this);
btn_sync_data.setOnClickListener(this);
} }
代码分析:
1、加载UI,初始化变量啥的跳过了,主要说一下关键代码,在第28代码中,与启动一个Activity类似,通过Intent想要启动的Service传递参数。
2、在37行通过bindService绑定Service,然后在ServiceConnection中获取Service类中onBind方法返回的实例,获取实例Service实例后,我们就可以通过调用Service中MyBinder的setData()方法对Service进行同步数据,如48行所示。
3、整个过程,在Service的onCreate方法中都会循环调用回调函数,同时我们在MainActivity中重写回调方法以实现更新UI。
四、测试
1、启动示例后,在输入框输入你好,然后点击startService,界面和对应的日志如下:
看了下面的代码后就会知道,此时因为没有绑定service,所以办法执行回调函数更新UI,所以显示区域没有更新。
2、点击bindService后,界面如下:
当执行bindService后,在ServiceConnection方法中就会执行执行回调函数更新UI,此时显示区域开始更新。
3、改变输入框内容,点击同步数据,界面和对应的日志如下:
因本人水平有限,如在文中发现错误或者描述不当的地方,敬请指正,感激不尽!
声明:欢迎转载,转载是请注明本文链接,谢谢!
Android学习笔记(九)一个例子弄清Service与Activity通信的更多相关文章
- Android学习笔记九:Service
一:Service是什么 Service,服务.一般用于提供需要在后台长期运行的服务(如复杂计算.下载等等耗时任务),其特点是长生命周期的.没有用户界面.在后台运行的. 二:Service的生命周期方 ...
- android学习笔记五。1、Service深入学习
一.Service,服务是没有界面而在后台长期运行的程序,可以看做是后台的Activity. 1.在Android中按返回键退出一个应用并不会(内存充足时)直接销毁一个进程,所以其中的子线程也可以在后 ...
- Android学习笔记_70_一个应用程序启动另一个应用程序的Activity
第一种(我自己写的) :之前在网上看来一些,很多不是我要的可以启动另外一个应用程序的主Activity. //这些代码是启动另外的一个应用程序的主Activity,当然也可以启动任意一个Activit ...
- Android学习笔记(八)深入分析Service启动、绑定过程
Service是Android中一个重要的组件,它没有用户界面,可以运行在后太做一些耗时操作.Service可以被其他组件启动,甚至当用户切换到其他应用时,它仍然可以在后台保存运行.Service 是 ...
- android学习笔记九——RatingBar
RatingBar==>星级评分条 RatingBar和SeekBar十分相似,它们甚至有相同的父类:AbsSeekBar.两者都允许用户通过拖动来改变进度: 两者最大的区别在于RatingBa ...
- Android学习笔记-构建一个可复用的自定义BaseAdapter
转载自http://www.runoob.com/w3cnote/android-tutorial-customer-baseadapter.html 作者:coder-pig 本节引言: 如题, ...
- Android学习笔记(5)----启动 Theme.Dialog 主题的Activity时程序崩溃的解决办法
新建了一个Android Studio工程,在MainActivity的主界面中添加了两个按钮,点击其中一个按钮用来启动 NormalActivity,点击另一按钮用来启动DialogActivity ...
- 【转】Pro Android学习笔记(十四):用户界面和控制(2):Text类控制
目录(?)[-] TextView 例子1在XML中设置autoLink属性 例子2在代码中设置autoLink属性 EditText AutoCompleteTextView MultiAutoCo ...
- 【转】 Pro Android学习笔记(九三):AsyncTask(2):小例子
目录(?)[-] 继承AsyncTask UI操作接口 使用AsyncTask 文章转载只能用于非商业性质,且不能带有虚拟货币.积分.注册等附加条件.转载须注明出处:http://blog.csdn. ...
随机推荐
- ERROR 1045 (28000): Access denied for user 'root'@'localhost'
# /etc/init.d/mysql stop# mysqld_safe --user=mysql --skip-grant-tables --skip-networking &# mysq ...
- Codevs 4560 NOIP2015 D2T2 子串
> 4560 NOIP2015 D2T2 子串 时间限制: 1 s 空间限制: 128000 KB 题目等级:黄金 Gold 题目描述 Description 有两个仅包含小写英文字母的字符串A ...
- OpenJudge 2775 文件结构“图”/ Poj 1057 FILE MAPPING
1.链接地址: http://bailian.openjudge.cn/practice/2775 http://poj.org/problem?id=1057 2.题目: 总时间限制: 1000ms ...
- 7种基本排序算法的Java实现
7种基本排序算法的Java实现 转自我的Github 以下为7种基本排序算法的Java实现,以及复杂度和稳定性的相关信息. 以下为代码片段,完整的代码见Sort.java 插入排序 /** * 直接插 ...
- js传带参数的函数
字符串: setTimeout('pageScroll(4)',100);
- win7 64位Apache http server+PHP配置
一.下载PHP 1.首先下载apache http server(我本来是想在官网下载,但是找半天也没找到,于是就在网上下了一个老版本的)我是在这个网址下载的:http://download.csdn ...
- C# 拷贝数组的几种方式
突然学到了,所以就放到博客上来共享一下,权当是学习日记吧. 首先说明一下,数组是引用类型的,所以注意不要在复制时复制了地址而没有复制数值哦! 其实在复制数组的时候,一定要用new在堆中开辟一块新的空间 ...
- 2016030208 - sql50题练习题
数据库建表脚本和使用的数据请参考:http://www.cnblogs.com/zhtzyh2012/p/5235826.html sql50题练习参看:http://blog.sina.com.cn ...
- warning: no newline at end of file
编译错误:warning: no newline at end of file原因:程序结尾需要有一个空行解决办法:在程序末尾多打个回车就行了
- Java 之 网络编程
1.OSI模型 a.全称:开放系统互联 b.七层模型:应用层.表示层.会话层.传输层.网络层.数据链路层.物理层 c.注意:OSI模型 是所谓的 理论标准 2.TCP/IP模型 a.四层模型:应用层. ...