一、实验目的
- 使同学掌握服务的隐式启动和显式启动方法。
- 使同学学会使用线程的启动、挂起和停止方法。
- 使同学学会跨线程的界面更新。
二、实验内容
- 使用“SimpleRandomServiceDemo”程序显式启动服务在应用程序中建立 Service?
- 使用“ThreadRandomServiceDemo”程序使用线程持续产生随机数?
- 使用“SimpleMathServiceDemo”程序使用绑定方式使用 Service?
- 使用“RemoteMathServiceDemo”程序,说明如何创建跨进程?
三、实验仪器、设备
- 硬件:PC 微型计算机、1G 以上内存,40G 以上硬盘
- 软件:Windows XP,Eclipse , JDK , Android SDK
四、实验步骤
前三个DEMO已经有了详尽代码,在此只展示代码不完全的第四个DEMO。
使用“RemoteMathServiceDemo”程序,说明如何创建跨进程?
- IMathService.aidl
// IMathService.aidl
package com.example.remotemathservicedemo;
// Declare any non-default types here with import statements
interface IMathService {
/**
* Demonstrates some basic types that you can use as parameters
* and return values in AIDL.
*/
long Add(long a,long b);
}
- MathService.java
package com.example.remotemathservicedemo;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.widget.Toast;
import androidx.annotation.Nullable;
public class MathService extends Service {
//实现接口中暴露给client的Stub,提供一个stub inner class来具体实现
private final IMathService.Stub mBinder = new IMathService.Stub(){
//具体实现AIDL文件中接口的定义的方法。
@Override
public long Add(long a, long b) {
return a + b;
}
};
@Nullable
@Override
//当client连接时,将触发onBind(),Service向client返回一个stub对象,因此client可以通过stub对象来访问Service。
public IBinder onBind(Intent intent) {
Toast.makeText(this, "远程绑定:MathService", Toast.LENGTH_SHORT).show();
return mBinder;
}
public boolean onUnbind (Intent intent){
Toast.makeText(this, "取消远程绑定:MathService", Toast.LENGTH_SHORT).show();
return false;
}
}
- MainActivity.java
package com.example.remotemathservicedemo;
import androidx.appcompat.app.AppCompatActivity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteException;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity {
private IMathService iMathService; //定义接口变量
private ServiceConnection connection; //定义连接变量
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
final TextView textView = (TextView)findViewById(R.id.label);
Button compute = (Button)findViewById(R.id.compute);
bindRemoteService();
compute.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
try{
long a = Math.round(Math.random()*100);
long b = Math.round(Math.random()*100);
//调用MathService提供的方法
long result = iMathService.Add(a,b);
String msg = String.valueOf(a) + " + " + String.valueOf(b) + " = " +String.valueOf(result);
textView.setText(msg);
}catch (RemoteException e){
e.printStackTrace();
}
}
});
}
private void bindRemoteService(){
//使用隐式Intent
final Intent intentService = new Intent("com.example.remotemathservicedemo.MathService");
intentService.setPackage("com.example.remotemathservicedemo");
connection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
//得到IMathService实例,可以调用IMathService里的函数
iMathService = IMathService.Stub.asInterface(service);
}
@Override
public void onServiceDisconnected(ComponentName name) {
iMathService = null;
}
};
//绑定远程服务
bindService(intentService,connection, Context.BIND_AUTO_CREATE);
}
}
- activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent"
>
<TextView android:id="@+id/label" android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="@string/hello">
</TextView>
<Button android:id="@+id/compute"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="加法运算" />
</LinearLayout>
- AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.remotemathservicedemo">
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<service
android:name=".MathService"
android:process=":remote"
>
<intent-filter>
<action android:name="com.example.remotemathservicedemo.MathService" />
</intent-filter>
</service>
</application>
</manifest>
五、实验截图
-
SimpleRandomServiceDemo
-
ThreadRandomServiceDemo
-
SimpleMathServiceDemo
-
RemoteMathServiceDemo
六、思考题
- 简述 Service 的原理和用途?
Service(服务)是一种能够在后台长期运行且不提供用户界面的应用程序组件。通常Service用来执行一些耗时操作,或者后台执行不提供用户交互界面的操作。另外一个应用程序组件可以启动一个Service,它可以在用户切换到其他应用的时候依旧保持在后台运行。另外,一个组件可以绑定到Service上,与它进行通信,甚至是IPC(进程间通信)。 - 如何通过 Service 实现 MP3 播放器音乐播放器?
把音乐播放相关操作写在一个Service类里面,然后在主活动中启动这个服务,这样当退出程序后,是由于该服务还在后台运行着,所以音乐还可以继续播放。