Android间的进程通讯(传递复杂对象)

Android间的进程通讯(传递复杂对象)

完成对复杂对象的序列化

在Android中传递复杂数据类型的时候要通过将序列化,在Android中提供了一个接口Parcelable来实现对对象的序列化。

下面对需要传输的对象进行序列化操作,首先看自定义的类Person。

package com.example.service_parcelable_conmmute.bean;

import android.graphics.Bitmap;
/**
 * 用来传输的对象结构
 * @author Xinyuyu
 *
 */
public class Person {
    private String name;
    private String age;
    private Bitmap figure;
    
    public Person(){
        
    }
    public Person(String name, String age, Bitmap figure){
        this.name = name;
        this.age = age;
        this.figure =figure;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getAge() {
        return age;
    }
    public void setAge(String age) {
        this.age = age;
    }
    public Bitmap getFigure() {
        return figure;
    }
    public void setFigure(Bitmap figure) {
        this.figure = figure;
    }
}

这个类就是一般的JavaBean包含了一些get/set方法,我们要做的就是在进程之间传递该对象的值。所以就要将该对象进行序列化,下面实现一个类来完成该任务。

建立ParcelablePerson类,来完成对Person的序列化和反序列化的操作。其代码如下:

public class ParcelablePerson implements Parcelable {
    Person person = new Person();
    
    public Person getPerson() {
        return person;
    }
    public void setPerson(Person person) {
        this.person = person;
    }
    public ParcelablePerson(Person person){
        this.person = person;
    }
    public ParcelablePerson(Parcel source){
        person.setName(source.readString());
        person.setAge(source.readString());
        person.setFigure((Bitmap)source.readParcelable(Bitmap.class.getClassLoader()));
    }
    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(person.getName());
        dest.writeString(person.getAge());
        dest.writeParcelable(person.getFigure(), PARCELABLE_WRITE_RETURN_VALUE);
    }
    
    public static final Parcelable.Creator<ParcelablePerson> CREATOR = new Parcelable.Creator<ParcelablePerson>(){

        @Override
        public ParcelablePerson createFromParcel(Parcel source) {
            return new ParcelablePerson(source);
        }
        @Override
        public ParcelablePerson[] newArray(int size) {
            return new ParcelablePerson[size];
        }
    };
}

这段代码中完成了对Person对象的序列化与反序列化,实现Parcelable接口要做的就是来重写几个方法(分别完成序列化和反序列化),writeToParcel(Parcel dest, int flags)完成的就是对对象的序列化

其中的参数Parcel是一个容器,将对象放入其中就是序列化的过程。而CREATOR中的createFormParcel(Parcel source)来完成的就是反序列化,可以返回一个复杂的对象。实现了这写,就完成了对复杂对象的序列化和反序列化操作,这样我们就可以对它进行传输了。

完成绑定服务,完成进程中的通讯

以上我们只是完成了对一个复杂对象的序列化,下面我们要做的是建立服务,并且完成传输。

首先我们要做的是创建一个服务,在Android中创建一个服务要做的就是去实现Service接口。下面是发送对象的一个服务代码

public class ServiceForSendObject extends Service {
    private static final String MSG = "MESSAGE";
    private ParcelablePerson parcelablePerson;
    private Person person;
    // =====================================
    SendObject.Stub sendBinder = new Stub(){

        @Override
        public ParcelablePerson getPersonInfo() throws RemoteException {
            Log.i(MSG, "调用getPersonInfo()接口实现,返回一个序列化的对象");
            return parcelablePerson;
        }};
    // =====================================
    
    @Override
    public void onCreate() {
        person = new Person("xinyuyu", "25", BitmapFactory.decodeStream(getResources().openRawResource(R.drawable.ic_launcher)));
        parcelablePerson = new ParcelablePerson(person);
        super.onCreate();
        Log.i(MSG, "回调onCreate()创建服务");
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.i(MSG, "回调onStartCommand()启动服务");
        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public boolean onUnbind(Intent intent) {
        Log.i(MSG, "回调onUnbind解绑服务");
        return super.onUnbind(intent);
    }

    @Override
    public void onDestroy() {
        Log.i(MSG, "回调onDestroy销毁服务");
        super.onDestroy();
    }

    @Override
    public IBinder onBind(Intent intent) {
        Log.i(MSG, "回调onBind()方法绑定服务");
        return sendBinder;
    }
}

在该段代码中输出各个阶段的一些信息,关键是在onCreat()方法中完成实例化一个对象,并且对该对象完成序列化操作。在分割线之间的代码是关键代码,注意在IBinder()中返回的是一个SendObject.Stub对象。而这个类是如何生成的呢?我们在完成进程间传输的时候需要用到Android中的AIDL语言来定义一个接口,下面我们来建立一个名为SendObject的aidl文件,SendObject.aidl

package com.example.service_parcelable_conmmute_service.aidl;

import com.example.service_parcelable_conmmute.bean.ParcelablePerson;

interface SendObject {
    ParcelablePerson getPersonInfo();
}

这个接口中的import会报错,这时候需要在建立一个AIDL文件,将其名字设置为和序列化类一样,这里就起做ParcelablePerson.aidl。其内容如下

package com.example.service_parcelable_conmmute.bean;

parcelable ParcelablePerson;

完成这些就不会有错误了。

这时候你会发现在gen目录下有一个和你的aidl名字一样Java文件即SendObject.java。而我们使用的SendObject.Stub就在其中。打开这个文件看一下,你会发现SendObject.Stub是这样一个类,实现了SendObject接口并且继承了android.os.Binder类。这样我们就重写SendObject方法,将其实现功能。就是分割线直接的代码。

public static abstract class Stub extends android.os.Binder 
implements com.example.service_parcelable_conmmute_service.aidl.SendObject

还要注意一点进程间的服务通讯应该更改AndroidManifest.xml文件的属性,如下

<service android:name="com.example.service_parcelable_conmmute.service.ServiceForSendObject"
android:process=":remote">
<intent-filter>
        <action android:name="com.example.service_parcelable_conmmute.service.SEND_OBJECT"/>
            <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>
</service>

这样可以完成隐式的传递Intent。完成了这些,服务端就完成了。这个时候可以运行下代码。

启动界面

点击初始化服务后,可以看到服务开始运行

红色框中为我们的服务。

点击两按钮后LogCat输出的信息

从中可以看出来这个服务的生命周期。

以上我们就完成了服务端的工作了,下面来进行客户端的完成。

在客户端我们首先需要的是将服务端写好的那些AIDL文件和实体类拷贝到该工程下(连同包名进行拷贝)如下图

然后我们所要完成的就是完成客户端的activity程序了。将俩个程序进行联系的就是我们的SendObject接口。看下面代码

public class A extends Activity {
    
    private Button show_button;
    private TextView show_name_age;
    private ImageView show_image;
    private Button unbind_ser;
    private SendObject sendObject;
    private ParcelablePerson parcelablePerson;
    private Person person;
    private ServiceConnection connection = new ServiceConnection(){
        
        // 建立绑定后service程序调用onBind()方法,返回一个IBinder对象
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            sendObject = SendObject.Stub.asInterface(service);    
            try {
                parcelablePerson = sendObject.getPersonInfo();
                person = parcelablePerson.getPerson();
                Log.i("", person.getName());    
                show_name_age.setText(person.getName() + "
" + person.getAge());
                show_image.setImageBitmap(person.getFigure());
                
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
            Log.i("", "error");            
        }
    };
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        
        
        super.onCreate(savedInstanceState);
        setContentView(R.layout.a);
        show_button = (Button)this.findViewById(R.id.button1);
        show_name_age = (TextView)this.findViewById(R.id.textView1);
        show_image = (ImageView)this.findViewById(R.id.imageView1);
        unbind_ser = (Button)this.findViewById(R.id.button2);
        
        show_button.setOnClickListener(new OnClickListener(){

            @Override
            public void onClick(View v) {
                // 隐式传递intent
                Intent intent = new Intent();
                intent.setAction("com.example.service_parcelable_conmmute.service.SEND_OBJECT");
                bindService(intent, connection, BIND_AUTO_CREATE);
            }});
        // 解除绑定(服务会在activity销毁后自动销毁)
        unbind_ser.setOnClickListener(new OnClickListener(){

            @Override
            public void onClick(View v) {
                unbindService(connection);  
            }
            
        });

    }
}

将该程序与服务进行绑定就用到了bindService()方法,看该方法中的参数,第一个是一个intent,第二个是一个ServiceConnection类,与服务建立连接,接收到onBind()传输的数据。从这个方法中我们就得到了传递过来的对象。运行客户端程序

点击第一个按钮后会显示出来一个Person对象的信息。这就说明数据从服务端传输了过来。

LogCat输出的信息

原文地址:https://www.cnblogs.com/xinyuyu/p/3718678.html