Activity跨进程调用其实很常见,比如在第三方sdk中的分享,授权登录等,如果我们有多个应用包相互调用共享activity 是要用到的。
1.新建项目,用于被调用 主activity 为 ReceiveActivity
2.注册activity,并且设置action
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".ReceiveActivity" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
<!--自定义action 用于可以跨进程被调用-->
<action android:name="com.test.MainActivity"/>
<!--自定义action 要添加-->
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
</activity>
</application>
特别注意,这里的action 用于调用该activity时的标签
3.ReceiveActivity 内代码实现
public class ReceiveActivity extends AppCompatActivity {
private static final String TAG = "ReceiveActivity";
private TextView mTest;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(activity_main);
initView();
}
private void initView() {
mTest = (TextView) findViewById(R.id.test);
Intent intent=getIntent();
if (intent!=null) {
String tag=intent.getStringExtra("tag");
if (tag!=null) {
mTest.setText("我收到了:"+tag);
}
}
}
}
xml
<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".ReceiveActivity">
<TextView
android:id="@+id/test"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</android.support.constraint.ConstraintLayout>
被调用端完毕
2.新建项目,用于调用ReceiveActivity 进程通讯
设置action并且传递参数
Intent intent = new Intent();
intent.setAction("com.test.ReceiveActivity");
intent.putExtra("tag", "调用成功了");
startActivity(intent);
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
/**
* 开始调用另一个进程
*/
private TextView mSend;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
}
private void initView() {
mSend = (TextView) findViewById(R.id.send);
mSend.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch (v.getId()) {
default:
break;
case R.id.send:
Intent intent = new Intent();
intent.setAction("com.test.MainActivity");
intent.putExtra("tag", "调用成功了");
startActivity(intent);
break;
}
}
}
xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity">
<TextView
android:id="@+id/send"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="30dp"
android:text="开始调用另一个进程"
/>
</LinearLayout>
将两个项目都安装到设备上,这时点击“开始调用另一个进程” 调起ReceiveActivity 并将传递的参数显示出来。
这里会发现启完成后查看进程,只有一个(先杀掉所有进程再启动),这是因为ReceiveActivity 使用的是默认的启动模式,重新new了一个ReceiveActivity 并且放到了调用项目的栈中,这样就达不到共享的目的,并且这时你打开被调用项目,发现不是刚才的那个了,又新建了一个ReceiveActivity 类,且相同类不在一个进程中。这里改下启动模式就好了
设置ReceiveActivity 启动模式
android:launchMode="singleInstance"
就这么多了,结束。