It is often handy to keep a service running in the background to manage certain tasks. They can be started from an Activity or a broadcast receiver. Managing the service lifecycle can be tricky unless you follow some key points, as I learned the hard way when creating Air Waves.
First off, services can either be;
- a) Bound by the activity lifecycle (created and destroyed along with activity). Normally services will be destroyed once they have no more work to do and no remaining activities are binded to them.
- b) Running along side with activity and only stops when it's done (or told to)
I'll show you how to start and manage the latter, a service that's running independently from an any binded Activities.
Defining
First you have to add some definition about the service in the AndroidManifest.xml file. Under application, simply add:
1.
<
service
android:name
=
".ContentCheckService"
></
service
>
That's it for the manifest!
Servicing
The service itself is a rather simple class.
01.
public
class
AirWavesService
extends
Service {
02.
// LocalBinder, mBinder and onBind() allow other Activities to bind to this service.
03.
public
class
LocalBinder
extends
Binder {
04.
public
AirWavesService getService() {
05.
return
AirWavesService.
this
;
06.
}
07.
}
08.
09.
private
final
LocalBinder mBinder =
new
LocalBinder();
10.
11.
@Override
12.
public
IBinder onBind(Intent intent) {
13.
return
mBinder;
14.
}
15.
16.
17.
// Variables
18.
protected
Handler handler;
19.
protected
Toast mToast;
20.
21.
22.
@Override
23.
public
void
onCreate() {
24.
super
.onCreate();
25.
26.
Log.i(
"Service"
,
"onCreate"
);
27.
28.
// Initialise UI elements
29.
handler =
new
Handler();
30.
mToast = Toast.makeText(
this
,
""
, Toast.LENGTH_SHORT);
31.
32.
// ...
33.
}
34.
35.
36.
@Override
37.
public
void
onDestroy() {
38.
Log.w(
"Service"
,
"onDestroy"
);
39.
// ...
40.
41.
// Clean up UI
42.
mToast =
null
;
43.
44.
super
.onDestroy();
45.
}
46.
47.
48.
@Override
49.
public
int
onStartCommand(Intent intent,
int
flags,
int
startId) {
50.
Log.i(
"Service"
,
"onStartCommand"
);
51.
return
android.app.Service.START_STICKY;
52.
}
53.
54.
55.
/**
56.
* Example function.
57.
* @throws AirWavesException
58.
*/
59.
public
void
doSomethingOnService()
throws
AirWavesException {
60.
if
(!isWiFiEnabled()) {
61.
throw
new
AirWavesException(
"No WiFi connection available."
);
62.
}
63.
64.
handler.post(
new
Runnable() {
65.
@Override
66.
public
void
run() {
67.
mToast.setText(
"do something"
);
68.
mToast.show();
69.
}
70.
});
71.
Log.i(
"Service"
,
"doSomethingOnService() called"
);
72.
}
73.
}
The value Service.START_STICKY returned by onStartCommand() controls the Service lifecycle. Basically it means that the service will be resurrected if it's killed by Android.
The documentation will explain this far better than I can.
Key functions
Shown above is the important parts of it. The main points you'll need are:
- onCreate() and onDestroy(): Use this to initialise and clean up your variables, much like in Activity.
- onStartCommand(): If you were writing a service which is bound to the activity lifecycle, then this is the fun one where you write your service logic. Since ours isn't, then we can write our functions anywhere, like in doSomethingOnService().
Note: if you're starting the service from a broadcast receiver then onStartCommand() will be called often, whereas onCreate() won't. You'll need to keep track of this yourself.
Starting the service
Now there's no point having a fancy service if you can't use it. To start it up, you'll need some simple yet (hopefully) familiar code:
1.
Intent i =
new
Intent(context, ContentCheckService.
class
);
2.
context.startService(i);
This is the key point you have to remember. If a service is started by startService(), then it will stay alive until you manually terminate it.
Connecting to the service
Once it's up and running, you can connect to the service and use it like a normal object.
The term you're looking for here is "binding" to a service. Services often provide functions which are needed by an Activity, and in order to access those functions easily you need to provide some "binding glue" between the two Activity and Service classes.
Within the Service class, notice the definition for LocalBinder. That allows your Activity to connect to the service via a ServiceConnection.
Here's a "trimmed down" version of the Activity class (I'm not kidding!)
01.
public
class
MainActivity
extends
Activity {
02.
protected
AirWavesService service;
03.
protected
AirWavesServiceConnection serviceConnection;
04.
05.
06.
@Override
07.
protected
void
onCreate(Bundle savedInstanceState) {
08.
super
.onCreate(savedInstanceState);
09.
10.
Log.i(
"Activity"
,
"onCreate"
);
11.
// ...
12.
}
13.
14.
@Override
15.
protected
void
onResume() {
16.
super
.onResume();
17.
Log.i(
"Activity"
,
"onResume"
);
18.
connectToService();
19.
}
20.
21.
@Override
22.
protected
void
onPause() {
23.
super
.onPause();
24.
25.
if
(serviceConnection !=
null
) {
26.
unbindService(serviceConnection);
27.
serviceConnection =
null
;
28.
}
29.
}
30.
31.
32.
// Helper function for connecting to AirWavesService.
33.
private
void
connectToService() {
34.
// Calling startService() first prevents it from being killed on unbind()
35.
startService(
new
Intent(
this
, AirWavesService.
class
));
36.
37.
// Now connect to it
38.
serviceConnection =
new
AirWavesServiceConnection();
39.
40.
boolean
result = bindService(
41.
new
Intent(
this
, AirWavesService.
class
),
42.
serviceConnection,
43.
BIND_AUTO_CREATE
44.
);
45.
46.
if
(!result) {
47.
throw
new
RuntimeException(
"Unable to bind with service."
);
48.
}
49.
}
50.
51.
52.
protected
class
AirWavesServiceConnection
implements
ServiceConnection {
53.
@Override
54.
public
void
onServiceConnected(ComponentName className, IBinder binder) {
55.
Log.i(
"Activity"
,
"onServiceConnected AirWavesService"
);
56.
service = ((AirWavesService.LocalBinder) binder).getService();
57.
}
58.
59.
@Override
60.
public
void
onServiceDisconnected(ComponentName className) {
61.
Log.e(
"Activity"
,
"onServiceDisconnected AirWavesService"
);
62.
service =
null
;
63.
}
64.
}
65.
66.
67.
protected
void
callServiceFunction()
throws
AirWavesException {
68.
service.doSomethingOnService();
69.
}
70.
71.
72.
@Override
73.
protected
void
onDestroy() {
74.
Log.e(
"Activity"
,
"onDestroy"
);
75.
76.
// If we no longer need it, kill the service
77.
if
(!G.isListening && !G.isSpeaking) {
78.
stopService(
new
Intent(
this
, AirWavesService.
class
));
79.
}
80.
81.
super
.onDestroy();
82.
}
83.
}
Taking a closer look
Alright working our way from top to bottom, you'll see "service" and "serviceConnection" declared at the top. They're important as they bind your Activity to the Service.
Nothing interesting is happening at onCreate().
However, just below is onResume() and onPause() which control the life of the binding connection.
Whenever your Activity is destroyed, the bind is no longer valid and needs to be undone. Rule of thumb is if the number of bindService() and unbindService() calls don't match up, you'll get debug error logs:
Activity has leaked ServiceConnection <X> that was originally bound here. android.app.ServiceConnectionLeaked
With connectToService(), the startService() call will kick-start it into a persistent service. By the time we call bindService(), we can be sure that the service is already up and running, ready for connection.
AirWavesServiceConnection is simply an implementation of ServiceConnection so we can get a handle to the service. In onServiceConnected(), we save the reference to the service and during onServiceDisconnected() we delete that reference.
This isn't really necessary, but callServiceFunction() shows you how to call functions from the service. Pretty easy eh?
And of course, onDestroy() stops the service using stopService() when we no longer need it.
And there you have it, a long life service on Android!
Just wait until you start reading into stuff like foreground services and wake-locks in order!
Sources
A mix of information from my previous post "Android: Detect when internet connectivity is connected" and documentation from the Cling library.
Big thanks to Mark Murphy the CommonsWare guy for hanging around StackOverflow and sharing so much of his knowledge.