Service
Service介绍
- Service 服务 是四大组件之一 和Activity非常相似
- 后台运行 没有界面
- 在清单文件中注册 都有自己的生命周期
Service特点
- Service 在后台运行 不用与用户进行交互 即使应用退出 服务也不会停止 当应用进程被杀死时 服务便会停止
- Service运行在主线程中 但需要执行耗时操作的时候 需要在服务中创建子线程完成
- Service的用途 播放音乐 后台下载大文件
Service的生命周期以及启动方式
1.启动方式:onCreate() — onStartCommand() — onDestroy()
开启服务:startService()
停止服务:在外部用stopService() 在内部用stopSelf()
注意:多次启动一个服务, 不会执行onCreate(), 每次都会执行onStartCommand()
2.绑定方式:onCreate() – onBind() — onUnbind() — onDestroy()
绑定服务:bindService()
解除绑定:unbindService()
注意:同一个服务, 只能被同一个启动源, 绑定一次, 除非解绑后, 可以再次绑定
3.先启动后绑定(常用): onCreate() – onStartCommand() – onBind() — onUnbind() – onDestroy()
onCreate() --> 服务被创建时调用
onStartCommand() --> 服务被启动时调用
onBind() --> 服务被绑定时调用
onUnbind() --> 服务解除绑定时调用
onDestroy() --> 服务被销毁时调用
startService 和 bindService的区别
start方式 | bind方式 | |
---|---|---|
和Activity的关系 | 一旦开启,长期后台运行,和Activity没有关系 | 如果开启者(Activity)退出了,服务也会跟着挂掉 |
是否可以调用服务中的方法 | 开启者Activity不能调用服务里的方法 | 开启者(Activity)可以间接的利用中间人调用服务里面的方法 |
如何创建Service
- 定义一个类 继承Service
- 重写父类的方法 onBind()
- 在清单文件中注册
start方式
MyService类
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.widget.Toast;
public class MyService extends Service {
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public void onCreate() {
super.onCreate();
Toast.makeText(this, "我是服务,我被创建了", Toast.LENGTH_SHORT).show();
}
@Override
public void onDestroy() {
super.onDestroy();
Toast.makeText(this, "我是服务,我被停止了", Toast.LENGTH_SHORT).show();
}
}
MainActivity类
import androidx.appcompat.app.AppCompatActivity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
public class MainActivity extends AppCompatActivity {
//开启服务的意图对象
private Intent intent;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
intent = new Intent(this,MyService.class);
}
public void start(View view) {
//开启服务
startService(intent);
}
public void stop(View view) {
//停止服务
stopService(intent);
}
}
效果图
startService效果图.gif
绑定方式
MyService类
import android.app.Service;
import android.content.Intent;
import android.os.Binder;
import android.os.IBinder;
import android.util.Log;
import android.widget.Toast;
public class MyService extends Service {
@Override
public IBinder onBind(Intent intent) {
Log.i("##", "onBind: ");
return new MyBinder();
}
class MyBinder extends Binder{
public MyService getService(){
return MyService.this;
}
}
public void toast(String message){
Toast.makeText(this, message, Toast.LENGTH_SHORT).show();
}
}
MainAvtivity类
import androidx.appcompat.app.AppCompatActivity;
import android.app.Service;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.util.Log;
import android.view.View;
public class MainActivity extends AppCompatActivity {
private Intent intent;
private MyService myService;
private ServiceConnection connection = new ServiceConnection() {
//服务连接时调用的方法
@Override
public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
myService = ((MyService.MyBinder) iBinder).getService();
Log.i("##", "onServiceConnected: ");
}
//异常断开连接时调用的方法
@Override
public void onServiceDisconnected(ComponentName componentName) {
Log.i("##", "onServiceDisconnected: ");
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
intent = new Intent(this, MyService.class);
}
public void bind(View view) {
//绑定服务
//Service.BIND_AUTO_CREATE 自动启动服务并绑定
bindService(intent,connection, Service.BIND_AUTO_CREATE);
}
public void unbind(View view) {
//解除绑定
unbindService(connection);
}
public void method(View view) {
myService.toast("Activity调用Service的方法");
}
}
效果图
bindService效果图.gif
前台服务
- 前台服务是用户知道 内存不足的时候不允许系统杀死的服务
- 前台必须给状态栏提供一个通知 它被放到正在运行的标题之下
- 只有在这个服务被终止或从前台主动移除通知后才能被解除
如果我们希望Service可以一直保持运行状态且不会在内存不足的情况下被回收时,可以选择将需要保持运行的Service设置为前台服务。
例如:音乐播放器
创建前台服务
MyService类
import android.app.Notification;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
public class MyService extends Service {
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
//通知点击意图
Intent intent1 = new Intent(getApplication(), MainActivity.class);
PendingIntent activity = PendingIntent.getActivity(getApplicationContext(), 2, intent1, PendingIntent.FLAG_UPDATE_CURRENT);
Notification build = new Notification.Builder(this)
.setSmallIcon(R.mipmap.ic_launcher)
.setContentTitle("通知")
.setContentText("我是前台服务")
.setContentIntent(activity)
.setAutoCancel(true)
.build();
//开始前台服务
startForeground(200, build);
return super.onStartCommand(intent, flags, startId);
}
@Override
public void onDestroy() {
super.onDestroy();
//停止前台服务
stopForeground(true);
}
}
MainActivity类
import androidx.appcompat.app.AppCompatActivity;
import android.app.Notification;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
public void click(View view) {
//开启服务
Intent intent = new Intent(this, MyService.class);
startService(intent);
}
}
效果图
前台服务效果图.gif
这样 一个简单的前台服务就写完了(忘记写关服务的方法了?)