[学习笔记]Android中Service通信

以下内容纯粹为本人学习笔记【记录】之用,所听课程(Q群群友百度网盘提供)为极客学院一位老师所讲(老师大名我尚未知晓),如有侵权请告知。在此特别感谢这位老师录制的视频资料。
1、启动Service并传递数据
MainAvtivity.java

public class MainActivity extends AppCompatActivity implements View.OnClickListener {
    private EditText etData;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        etData = (EditText) findViewById(R.id.etData);

        //创建监听器
        findViewById(R.id.btnStartSvc).setOnClickListener(this);
        findViewById(R.id.btnStopSvc).setOnClickListener(this);
    }


    @Override
    public void onClick(View view) {
        switch (view.getId()) {
            case R.id.btnStartSvc:
                Intent i = new Intent(this, MyService.class);
                i.putExtra("data", etData.getText().toString());//将数据快递过来
                startService(i);
                break;

            case R.id.btnStopSvc:
                stopService(new Intent(this, MyService.class));
                break;
        }
    }
}

MyService.java

public class MyService extends Service {
    private boolean running =false;
    private String data = "这是默认信息!!form Service";

    public MyService() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        throw new UnsupportedOperationException("Not yet implemented");
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        //Service接收数据
        data = intent.getStringExtra("data");

        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public void onCreate() {
        super.onCreate();

        running = true;
        new Thread(){
            @Override
            public void run() {
                super.run();

                while (running) {
                    System.out.println(data);
                    try {
                        sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }.start();
    }
    @Override
    public void onDestroy() {
        super.onDestroy();
        running =false;
    }
}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.keen.connectservice.MainActivity">

    <EditText
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="默认信息 from AndroidDev"
        android:id="@+id/etData"/>

    <Button
        android:text="启动服务"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/btnStartSvc" />

    <Button
        android:text="停止服务"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/btnStopSvc" />
</LinearLayout>

效果
这里写图片描述
2、绑定Service进行通信
布局文件添加两个按钮,分别用于绑定和解除服务

    <Button
        android:text="绑定服务"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/btnBindSvc" />

    <Button
        android:text="解除绑定服务"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/btnUnBindSvc" />

同时MainActivity添加对应的监听事件

findViewById(R.id.btnBindSvc).setOnClickListener(this);
        findViewById(R.id.btnUnBindSvc).setOnClickListener(this);

并添加对应的点击事件

 case R.id.btnBindSvc:
                bindService(new Intent(this, MyService.class), this, Context.BIND_AUTO_CREATE);
                break;

            case R.id.btnUnBindSvc:
                unbindService(this);
                break;

以及重载两个办法,让MainActivity实现ServiceConnected

    @Override
    public void onServiceConnected(ComponentName componentName, IBinder iBinder) {

    }

    @Override
    public void onServiceDisconnected(ComponentName componentName) {

    }

再添加一个同步数据的按钮,即输入文本框里的数据同步到Service

<Button
        android:text="同步数据"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/btnSyncData" />

当然同时也添加监听事件和点击事件:

findViewById(R.id.btnSyncData).setOnClickListener(this);

点击事件里如何实现同步数据操作?Service提供了一个办法,通过IBinder对象来实现Activity和Service两个基本组件之间的连接(如何连接的?MainActivity中的onServiceConnected()中的参数IBinder iBinder即可访问到MyService中onBind()的返回值,即Binder实例)。
在MyService.java创建一个公开类,且继承自系统(os)的Binder,并将onBind()修改,代码如下:

    @Override
    public IBinder onBind(Intent intent) {
        return new Binder();
    }
    public class Binder extends android.os.Binder {

    }

在类内部写方法

    public class Binder extends android.os.Binder {
        public void setData(String data) {
            //通过这里的方法即可修改Service中data的值
            MyService.this.data = data;
        }
    }

此时在Service外部,即Activity里即可操作。首先定义一个Binder对象

private MyService.Binder binder = null;

之后在onServiceConnected()

    @Override
    public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
        binder = (MyService.Binder) service;//强制转化类型
    }

添加同步按钮的点击事件方法

            case R.id.btnSyncData:
                if (binder!=null) {
                    binder.setData(etData.getText().toString());//数据来自etData
                }
                break;

通过这种方式,直接办法调用把数据传给Service,即通过binder进行通信。相比通过startService()方式通信更加方便高效。

如何监听服务的内部状态?如Service发生了改变时,如何通知给外界代码。实例:Servicie内部信息呈现到外界。
修改MyService的循环代码

        new Thread(){
            @Override
            public void run() {
                super.run();

                int i = 0;

                while (running) {

                    i++;

                    System.out.println(i+ ":" +data);//除了在控制输出外,如何呈现到Activity的TextView里
                    try {
                        sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }.start();

除了在控制输出外,如何呈现到Activity的TextView里。
创建一个TextView

    <TextView
        android:text="TextView"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/tvOut" />

MainActivity声明一个tvout

private TextView tvout;//定义/声明
tvOut = (TextView) findViewById(R.id.tvOut);

接下来的操作是:如何通知外界,即回调
MyService写一个Callback回调接口、callback变量、setCallback方法、getCallback方法

    private Callback callback = null;//变量赋值为null

    public void setCallback(Callback callback) {
        this.callback = callback;
    }
    public Callback getCallback() {
        return callback;
    }

    public static interface Callback {
        void onDataChange(String data);
    }

修改循环语句如

while (running) {

                    i++;
                    String str = i+ ":" +data;

                    System.out.println(str);//除了在控制输出外,如何呈现到Activity的TextView里
                    if (callback!=null) {
                        callback.onDataChange(str);//传出str
                    }
                    try {
                        sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }

服务外界如何添加事件的绑定?在MyService中Binder类写一个办法:

public MyService getService() {
            return MyService.this;
        }

此时外界(Activity)就可以访问到(Service)。
修改MainActivity

    @Override
    public void onServiceConnected(ComponentName componentName, IBinder iBinder) {

//        binder = (MyService.Binder) service;//强制类型转换
        binder.getService().setCallback(new MyService.Callback() {
            @Override
            public void onDataChange(String data) {

            }
        });
    }

因为在Android中有 一个安全机制,UI线程是不允许其他辅线程来直接修改UI线程的资源。因此,定义Handle,并重写handleMessage()方法。

    private Handler handler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
        }
    };

再次修改MainActivity中onDataChange()方法

public void onDataChange(String data) {
                Message msg = new Message();//创建
                Bundle b = new Bundle();
                b.putString("data", data);
                msg.setData(b);//附加数据
                handler.sendMessage(msg);//传递数据

            }

获取到msg之后,在handleMessage()获取到字符串,并填入tvOut.setText()中,这样就能呈现到tvOut文本框中了。

        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            tvOut.setText(msg.getData().getString("data"));
        }

点击[绑定服务],效果:
这里写图片描述

源码完整部分
MainActivity

public class MainActivity extends AppCompatActivity implements View.OnClickListener, ServiceConnection {
    private EditText etData;
    private MyService.Binder binder =null;
    private TextView tvOut;//定义/声明

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        etData = (EditText) findViewById(R.id.etData);
        tvOut = (TextView) findViewById(R.id.tvOut);

        //创建监听器
        findViewById(R.id.btnStartSvc).setOnClickListener(this);
        findViewById(R.id.btnSyncData).setOnClickListener(this);
        findViewById(R.id.btnBindSvc).setOnClickListener(this);
        findViewById(R.id.btnUnBindSvc).setOnClickListener(this);
        findViewById(R.id.btnStopSvc).setOnClickListener(this);
    }


    @Override
    public void onClick(View view) {
        switch (view.getId()) {
            case R.id.btnStartSvc:
                Intent i = new Intent(this, MyService.class);
                i.putExtra("data", etData.getText().toString());//将数据快递过来
                startService(i);
                break;

            case R.id.btnSyncData:
                if (binder!=null) {
                    binder.setData(etData.getText().toString());//数据来自etData
                }
                break;

            case R.id.btnBindSvc:
                bindService(new Intent(this, MyService.class), this, Context.BIND_AUTO_CREATE);
                break;

            case R.id.btnUnBindSvc:
                unbindService(this);
                break;

            case R.id.btnStopSvc:
                stopService(new Intent(this, MyService.class));
                break;
        }
    }

    @Override
    public void onServiceConnected(ComponentName componentName, IBinder iBinder) {

        binder = (MyService.Binder) service;//强制类型转换
        binder.getService().setCallback(new MyService.Callback() {
            @Override
            public void onDataChange(String data) {
                Message msg = new Message();//创建
                Bundle b = new Bundle();
                b.putString("data", data);
                msg.setData(b);//附加数据
                handler.sendMessage(msg);//传递数据

            }
        });
    }
    @Override
    public void onServiceDisconnected(ComponentName componentName) {
    }

    private Handler handler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            tvOut.setText(msg.getData().getString("data"));
        }
    };
}

MyService

public class MyService extends Service {
    private boolean running =false;
    private String data = "这是默认信息!!form Service";

    public MyService() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        return new Binder();
    }
    public class Binder extends android.os.Binder {
        public void setData(String data) {
            //通过这里的方法即可修改Service中data的值
            MyService.this.data = data;
        }
        public MyService getService() {
            return MyService.this;
        }
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        //Service接收数据
        data = intent.getStringExtra("data");

        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public void onCreate() {
        super.onCreate();

        running = true;
        new Thread(){
            @Override
            public void run() {
                super.run();

                int i = 0;

                while (running) {

                    i++;
                    String str = i+ ":" +data;

                    System.out.println(str);//除了在控制输出外,如何呈现到Activity的TextView里
                    if (callback!=null) {
                        callback.onDataChange(str);//传出str
                    }
                    try {
                        sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }.start();
    }
    @Override
    public void onDestroy() {
        super.onDestroy();
        running =false;
    }

    private Callback callback = null;//变量赋值为null

    public void setCallback(Callback callback) {
        this.callback = callback;
    }
    public Callback getCallback() {
        return callback;
    }

    public static interface Callback {
        void onDataChange(String data);
    }
}

activity_main

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.keen.connectservice.MainActivity">

    <TextView
        android:text="TextView"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/tvOut" />

    <EditText
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="默认信息 from AndroidDev"
        android:id="@+id/etData"/>

    <Button
        android:text="启动服务"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/btnStartSvc" />

    <Button
        android:text="同步数据"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/btnSyncData" />

    <Button
        android:text="绑定服务"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/btnBindSvc" />

    <Button
        android:text="解除绑定服务"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/btnUnBindSvc" />

    <Button
        android:text="停止服务"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/btnStopSvc" />
</LinearLayout>
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值