Android跨进程通信访问其他应用程序的Activity

访问其他应用程序的Activity
Activity既可以在进程内(同一个应用程序)访问,也可以跨进程访问。如果想在同一个应用程序中访问Activity,需要指定Context对象和Activity的Class对象,代码如下:

Intent intent = new Intent(this, Test.class);
startActivity(intent);  

Activity的跨进程访问与进程内访问略有不同。虽然它们都需要Intent对象,但跨进程访问并不需要指定Context对象和Activity的 Class对象,而需要指定的是要访问的Activity所对应的Action(一个字符串)。有些Activity还需要指定一个Uri(通过 Intent构造方法的第2个参数指定)。在android系统中有很多应用程序提供了可以跨进程访问的Activity,例如,下面的代码可以直接调用拨打电话的Activity。

Intent callIntent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:12345678");
startActivity(callIntent);  

在调用拨号程序的代码中使用了一个Intent.ACTION_CALL常量,该常量的定义如下:

public static final String ACTION_CALL = "android.intent.action.CALL";

这个常量是一个字符串常量,也是我们在这节要介绍的跨进程调用Activity的关键。如果在应用程序中要共享某个Activity,需要为这个 Activity指定一个字符串ID,也就是Action。也可以将这个Action看做这个Activity的key。在其他的应用程序中只要通过这个 Action就可以找到与Action对应的Activity,并通过startActivity方法来启动这个Activity。

下面先来看一下如何将应用程序的Activity共享出来,读者可按如下几步来共享Activity:
1. 在AndroidManifest.xml文件中指定Action。指定Action要使用<action>标签,并在该标签的android:name属性中指定Action
2. 在AndroidManifest.xml文件中指定访问协议。在指定Uri(Intent类的第2个参数)时需要访问协议。访问协议需要使 用<data>标签的android:scheme属性来指定。如果该属性的值是“abc”,那么Uri就应该是“abc://Uri的主体 部分”,也就是说,访问协议是Uri的开头部分。
3. 通过getIntent().getData().getHost()方法获得协议后的Uri的主体部分。这个Host只是个称谓,并不一定是主机名。读者可以将其看成是任意的字符串。
4. 从Bundle对象中获得其他应用程序传递过来的数据。
5. 这一步当然是获得数据后做进一步的处理了。至于如何处理这些数据,就得根据具体的需求决定了。
下面来根据这些步骤共享一个Activity。首先建立一个android工程(ActionActivity),工程的主Activity是Main。在 本例中我们会共享这个Main类。首先打开AndroidManifest.xml文件,添加一个<activity>标签,并重新定义了 Main的相应属性。AndroidManifest.xml文件的内容如下:

<!--  重新配置Main  -->
<activity android:name=".Main"  android:label="@string/app_name" >
    <intent-filter>
        <action android:name="net.blogjava.mobile.MYACTION"  />
        <data android:scheme="info"  />
        <category android:name="android.intent.category.DEFAULT"  />
    </intent-filter>
</activity>  

在配置AndroidManifest.xml时要注意,不能在同一个<activity>中配置多个动作,否则会覆盖MAIN动作以使该程序无法正常启动(虽然其他应用程序调用Main是正常的)。从上面的代码可以看出,<action>标签的android:name属性值是 net.blogjava.mobile.MYACTION,这就是Main自定义的动作。<data>标签指定了Url的协议。如果指定 了<data>标签的android:scheme属性值(info),则在调用Main时需要使用如下的URL: info://任意字符串

一般<category>标签的android:name属性值可以设成android.intent.category.DEFAULT。

下面来看看如何在Main类的onCreate方法中获得其他应用程序传递过来的数据。

public class Main extends Activity implements OnClickListener {
    private EditText editText;

    @Override
    public void onClick(View view) {
        // 单击按钮,会显示文本框中的内容(以Toast信息框形式显示)
        Toast.makeText(this, editText.getText().toString(), Toast.LENGTH_LONG).show();
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        Button button = (Button) findViewById(R.id.button);
        button.setOnClickListener(this);
        editText = (EditText) findViewById(R.id.edittext);
        // 获得其他应用程序传递过来的数据
        if (getIntent().getData() != null) {
            // 获得Host,也就是info://后面的内容
            String host = getIntent().getData().getHost();
            Bundle bundle = getIntent().getExtras();
            // 其他的应用程序会传递过来一个value值,在该应用程序中需要获得这个值
            String value = bundle.getString("value");
            // 将Host和Value组合在一下显示在EditText组件中
            editText.setText(host + ":" + value);
            // 调用了按钮的单击事件,显示Toast信息提示框
            onClick(button);
        }
    }
}

从上面的程序可以看出,首先通过getIntent().getData()来判断其他的应用程序是否传递了Uri(getData方法返回了一个Uri 对象)。如果运行该程序,Uri为null,因此,不会执行if语句里面的代码。当其他的应用程序传递了Uri对象后,系统会执行if语句里面的代码。当 运行ActionActivity后,在文本框中输入“Running”,单击“显示文本框的内容”按钮,会显示如图2所示的Toast提示信息框。

下面来看一下其他的应用程序是如何调用ActionActivity中的Main。新建一个android工程(InvokeActivity),并添加一个按钮,按钮的单击事件方法代码如下:

public void onClick(View view) {
        // 需要使用Intent类的第2个参数指定Uri
        Intent intent = new Intent("net.blogjava.mobile.MYACTION", Uri.parse("info://调用其他应用程序的Activity"));
        // 设置value属性值
        intent.putExtra("value", "调用成功");
        // 调用ActionActivity中的Main
        startActivity(intent);
}

在运行InvokeActivity之前,先要运行ActionActivity以便在android模拟器中安装该程序。然后单击InvokeActivity中的按钮,就会显示如图3所示的效果。

当然,也可以使用startActivityForResult方法来启动其他应用程序的Activity,以便获得Activity的返回值。例如,可以将ActionActivity中Main类的onClick代码修改为下面的形式。

    public void onClick(View view) {
        Toast.makeText(this, editText.getText().toString(), Toast.LENGTH_LONG).show();
        Intent intent = new Intent();
        // 设置要返回的属性值
        intent.putExtra("result", editText.getText().toString());
        // 设置返回码和Intent对象
        setResult(2, intent);
        // 关闭Activity
        finish();
    }

然后在InvokeActivity中使用下面的代码来调用Main。

        intent = new Intent("net.blogjava.mobile.MYACTION", Uri.parse("info://调用其他应用程序的Activity"));
        // 传递数据
        intent.putExtra("value", "调用成功");
        startActivityForResult(intent, 1); // 1为请求码

要想接收Activity返回的值,需要覆盖onActivityResult事件方法,代码如下:

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        Toast.makeText(this, "返回值:" + data.getExtras().getString("result"), Toast.LENGTH_LONG).show();
    }

当单击InvokeActivity中的相应按钮后,并且Main关闭后,会显示如图4所示的Toast信息提示框。

从本节的介绍可以看出,跨进程访问Activity(访问其他应用程序中的Activity)主要是通过一个Action来完成的,如果要传递数据,还需 要指定一个Uri。当然,传递数据也可以通过Intent来完成。传递数据的过程可以是双向的。如果要想从调用的Activity中返回数据,就需要使用 startActivityForResult方法来启动Activity了。

原文地址:https://www.cnblogs.com/chenxibobo/p/9648744.html

时间: 2024-12-13 13:18:45

Android跨进程通信访问其他应用程序的Activity的相关文章

【朝花夕拾】一篇文章搞懂Android跨进程通信

前言 只要是面试中高级工程师岗位,Android跨进程通信就是最受面试官青睐的知识点.Android系统的运行由大量相互独立的进程相互协助来完成的,所以Android进程间通信问题,是做好Android开发高级工程师必须要跨过的一道坎.如果您还对这方面的知识还做不到如数家珍,那就和我一起来攻克它吧! 本文主要包含了如下内容: 其行文脉络大致如下,希望能加深读者对这方面内容的记忆:(1)Android基于Linux系统,所以先说系统进程相关知识和Linux IPC.(2)总结Android的IPC

【朝花夕拾】性能优化篇之(八)AIDL与Android跨进程通信

一.Linux进程间通信 1.进程隔离 在操作系统中,进程与进程间的内存和数据都是不共享的.两个进程就好像大海中相互独立的两个岛屿,各自生活在互相平行的两个世界中,互不干扰,各自为政.这样做的目的,是为了避免进程间相互操作数据的现象发生,从而引起各自的安全问题.为了实现进程隔离,采用了虚拟地址空间,两个进程各自的虚拟地址不同,从逻辑上来实现彼此间的隔离. 马克思主义哲学说,人是一切社会关系的总和.任何一个个体都不可能完全隔离于外界,都不可避免地与外界"互通有无".进程也一样,时不时需要

android 跨进程通信

转自:http://www.androidsdn.com/article/show/137 由于android系统中应用程序之间不能共享内存.因此,在不同应用程序之间交互数据(跨进程通讯)就稍微麻烦一些.在android SDK中提供了4种用于跨进程通讯的方式.这4种方式正好对应于android系统中4种应用程序组件:Activity.Content Provider.Broadcast和Service. 其中Activity可以跨进程调用其他应用程序的Activity: Content Pro

*Android跨进程通信的四种方式

由于android系统中应用程序之间不能共享内存.因此,在不同应用程序之间交互数据(跨进程通讯)就稍微麻烦一些.在android SDK中提供了4种用于跨进程通讯的方式.这4种方式正好对应于android系统中4种应用程序组件:Activity.Content Provider.Broadcast和Service.其中Activity可以跨进程调用其他应用程序的Activity:Content Provider可以跨进程访问其他应用程序中的数据(以Cursor对象形式返回),当然,也可以对其他应

Android跨进程通信

由于android系统中应用程序之间不能共享内存.因此,在不同应用程序之间交互数据(跨进程通讯)就稍微麻烦一些.在android SDK中提供了4种用于跨进程通讯的方式.这4种方式正好对应于android系统中4种应用程序组件:Activity.Content Provider.Broadcast和Service. 其中Activity可以跨进程调用其他应用程序的Activity: Content Provider可以跨进程访问其他应用程序中的数据(以Cursor对象形式返回),当然也可以对其他

Android跨进程通信AIDL服务

服务(Service)是android系统中非常重要的组件.Service可以脱离应用程序运行.也就是说,应用程序只起到一个启动Service的作用.一但Service被启动,就算应用程序关闭,Service仍然会在后台运行. android系统中的Service主要有两个作用:后台运行和跨进程通讯.后台运行就不用说了,当Service启动后,就可以在Service对象中 运行相应的业务代码,而这一切用户并不会察觉.而跨进程通讯是这一节的主题.如果想让应用程序可以跨进程通讯,就要使用我们这节讲的

Android跨进程通信Messenger

一.概述 我们可以在客户端发送一个Message给服务端,在服务端的handler中会接收到客户端的消息,然后进行对应的处理,处理完成后,再将结果等数据封装成Message,发送给客户端,客户端的handler中会接收到处理的结果. 有这么几个特点: 基于Message,相信大家都很熟悉 支持回调的方式,也就是服务端处理完成长任务可以和客户端交互 不需要编写aidl文件 此外,还支持,记录客户端对象的Messenger,然后可以实现一对多的通信:甚至作为一个转接处,任意两个进程都能通过服务端进行

【朝花夕拾】跨进程通信,你只知道AIDL,就OUT了

一.前言 转载请声明,转自[https://www.cnblogs.com/andy-songwei/p/11774836.html],谢谢! 提起跨进程通信,大多数人首先会想到AIDL.我们知道,用AIDL来实现跨进程通信,需要在客户端和服务端都添加上aidl文件,并在服务端的Service中实现aidl对应的接口.如果还需要服务端给客户端发送信息,还需要再添加回调相关的aidl文件,以及使用RemoteCallbackList来辅助实现该功能.在我的另外一篇文章[朝花夕拾]Android性能

android 远程Service以及AIDL的跨进程通信

在Android中,Service是运行在主线程中的,如果在Service中处理一些耗时的操作,就会导致程序出现ANR. 但如果将本地的Service转换成一个远程的Service,就不会出现这样的问题了. 转换成远程Service非常简单,只需要在注册Service的时候将他的android:process的属性制定成 :remote就可以了. 重新运行项目,你会发现,不会出现ANR了. 为什么将MyService转换成远程Service后就不会导致程序ANR了呢?这是由于,使用了远程Serv