我发现了一种简单而优雅的方法:
方法一
第一次活动的代码:
final Object objSent = new Object();
final Bundle bundle = new Bundle();
bundle.putBinder("object_value", new ObjectWrapperForBinder(objSent));
startActivity(new Intent(this, SecondActivity.class).putExtras(bundle));
Log.d(TAG, "original object=" + objSent);
第二项活动的代码:
final Object objReceived = ((ObjectWrapperForBinder)getIntent().getExtras().getBinder("object_value")).getData();
Log.d(TAG, "received object=" + objReceived);
您会发现objSent
&objReceived
相同hashCode
,因此它们是相同的。
但是为什么我们可以通过这种方式传递一个Java对象呢?
实际上,Android绑定器将为Java对象创建全局JNI引用,并在没有Java对象的引用时释放该全局JNI引用。活页夹会将这个全局JNI引用保存在Binder对象中。
*警告:除非两个活动在同一进程中运行,否则此方法仅适用,否则将在(ObjectWrapperForBinder)getIntent()。getExtras()。getBinder(“ object_value”)抛出ClassCastException *
类ObjectWrapperForBinder定义
public class ObjectWrapperForBinder extends Binder {
private final Object mData;
public ObjectWrapperForBinder(Object data) {
mData = data;
}
public Object getData() {
return mData;
}
}
方法二
- 对于发送者,
- 使用自定义本地方法将Java对象添加到JNI全局引用表(通过JNIEnv :: NewGlobalRef)
- 将返回整数(实际上是JNIEnv :: NewGlobalRef返回jobject,它是一个指针,我们可以安全地将其强制转换为int)放入您的Intent(通过Intent :: putExtra)
- 给接收者
- 从Intent获取整数(通过Intent :: getInt)
- 使用自定义本机方法从JNI全局引用表还原Java对象(通过JNIEnv :: NewLocalRef)
- 从JNI全局引用表中删除项目(通过JNIEnv :: DeleteGlobalRef),
但是方法2有一个严重的问题,如果接收方无法还原Java对象(例如,在还原Java对象之前发生了一些异常,或者接收方Activity根本不存在),则Java对象将成为孤立或内存泄漏,方法1没有此问题,因为android绑定程序将处理此异常
方法3
要远程调用java对象,我们将创建一个数据协定/接口来描述java对象,我们将使用aidl文件
IDataContract.aidl
package com.example.objectwrapper;
interface IDataContract {
int func1(String arg1);
int func2(String arg1);
}
第一次活动的代码
final IDataContract objSent = new IDataContract.Stub() {
@Override
public int func2(String arg1) throws RemoteException {
// TODO Auto-generated method stub
Log.d(TAG, "func2:: arg1=" + arg1);
return 102;
}
@Override
public int func1(String arg1) throws RemoteException {
// TODO Auto-generated method stub
Log.d(TAG, "func1:: arg1=" + arg1);
return 101;
}
};
final Bundle bundle = new Bundle();
bundle.putBinder("object_value", objSent.asBinder());
startActivity(new Intent(this, SecondActivity.class).putExtras(bundle));
Log.d(TAG, "original object=" + objSent);
第二项活动的代码:
将AndroidManifest.xml中的android:process属性更改为非空进程名称,以确保第二个活动在另一个进程中运行
final IDataContract objReceived = IDataContract.Stub.asInterface(getIntent().getExtras().getBinder("object_value"));
try {
Log.d(TAG, "received object=" + objReceived + ", func1()=" + objReceived.func1("test1") + ", func2()=" + objReceived.func2("test2"));
} catch (RemoteException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
这样,即使两个活动在不同的进程中运行,我们也可以在它们之间传递接口,并远程调用该接口方法
方法4
方法3似乎还不够简单,因为我们必须实现一个aidl接口。如果您只是想做简单的任务而方法的返回值是不必要的,我们可以使用android.os.Messenger
第一次活动的代码(发送者):
public class MainActivity extends Activity {
private static final String TAG = "MainActivity";
public static final int MSG_OP1 = 1;
public static final int MSG_OP2 = 2;
public static final String EXTRA_MESSENGER = "messenger";
private final Handler mHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
// TODO Auto-generated method stub
Log.e(TAG, "handleMessage:: msg=" + msg);
switch (msg.what) {
case MSG_OP1:
break;
case MSG_OP2:
break;
default:
break;
}
super.handleMessage(msg);
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
startActivity(new Intent(this, SecondActivity.class).putExtra(EXTRA_MESSENGER, new Messenger(mHandler)));
}
}
第二项活动的代码(接收者):
public class SecondActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_second);
final Messenger messenger = getIntent().getParcelableExtra(MainActivity.EXTRA_MESSENGER);
try {
messenger.send(Message.obtain(null, MainActivity.MSG_OP1, 101, 1001, "10001"));
messenger.send(Message.obtain(null, MainActivity.MSG_OP2, 102, 1002, "10002"));
} catch (RemoteException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
所有Messenger.send将在Handler中异步并顺序执行。
实际上,android.os.Messenger也是一个辅助界面,如果您拥有android源代码,则可以找到一个名为IMessenger.aidl的文件
package android.os;
import android.os.Message;
/** @hide */
oneway interface IMessenger {
void send(in Message msg);
}