一、ComponentName简介

ComponentName,顾名思义,就是组件名称,这个类主要用来定义一个应用程序的组件,通过调用Intent中的setComponent方法,我们可以打开同个应用以及不同应用中的组件。例如:Activity,Service。

二、ComponentName的使用

简单介绍完ComponentName,接下来来看看如何使用它。由于ComponentName实现了Parcelable接口,所以它可以实现跨进程通信。要使用ComponentName,我们需要了解它的构造函数。实例化一个ComponentName需要两个参数,第一个参数是要启动应用的包名称getPackageName或者上下文Context,这个包名称是指清单文件中列出的应用的包名称。

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.hjr.heidoulib"  //就是这里
    >
    <application

第二个参数是你要启动的Activity或者Service的全称(包名+类名)。它的构造函数有三种,如下:

1、 第一种
/**
*@param pkg 组件存在的包的名称。该值不能为空,例如:"com.test.componentName"
* @param cls 需要在pkg 中的类的名称 例如:"com.test.componentName.MainActivity"
*/
public ComponentName(@NonNull String pkg, @NonNull String cls) {
        ......
    }
2、第二种
/**
* @param pkg 实现组件的包的上下文 例如:MainActivity.this
* @param cls 需要在pkg 中的类的名称 例如:"com.test.componentName.MainActivity"
* 或者MainActivity.class.getName()
*/
public ComponentName(@NonNull Context pkg, @NonNull String cls) {
        ......
    }
3、第三种
/**
* @param pkg 实现组件的包的上下文 例如:MainActivity.this
* @param cls 所需组件的类对象 例如:TestActivity.class
*/
public ComponentName(@NonNull Context pkg, @NonNull Class<?> cls) {
        ......
    }

现在来写一个完成的实例。
首先是布局,很简单,只要实现一个开启Intent的按钮。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="TestComponentName"
        android:onClick="startTestComponentName"
        />
</LinearLayout>

接下来是写开启对应组件的逻辑,这里写了开启Activity以及Service两种组件。如下:

/**
     * 开始ComponentName测试
     */
    public void startTestComponentName(View view){
        Intent intent = new Intent(); //创建Intent对象
        String pageName = getPackageName();//获取应用
        //第一种方式实现
        ComponentName componentName1 = new ComponentName(pageName,"com.hjr.heidoulib.module.activity.TestComponentNameActivity");
        //第二种方式实现
        ComponentName componentName2 = new ComponentName(MainActivity.this,"com.hjr.heidoulib.module.activity.TestComponentNameActivity");
        //第三种方式实现(实现的是Service)
        ComponentName componentName3 = new ComponentName(MainActivity.this,JobTestService.class);
        intent.setComponent(componentName1);//调用Intent的setComponent()方法实现传递
        startActivity(intent);//显示启动Activity
    }

这样子你就可以轻松的实现Intent+ComponentName来打开应用中不同的组件了。

三、需要注意的点。

如果你要的启动的其他应用的Activity不是该应用的入口Activity,那么在清单文件中,该Activity节点一定要加上android:exported=”true”,表示允许其他应用打开,对于所有的Service,如果想从其他应用打开,也都要加上这个属性:

<activity android:name=".module.activity.TestComponentNameActivity"
            android:label="TestComponentName"
            android:exported="true" //注意></activity>

        <service
            android:name=".module.service.JobTestService"
            android:enabled="true"
            android:exported="true"//注意
            android:permission="android.permission.BIND_JOB_SERVICE" />

对于除了入口Activity之外的其他组件,如果不加这个属性,都会抛出“java.lang.SecurityException: Permission Denial…..”异常

那么为什么入口Activity不用添加这个属性就可以被其他应用启动呢?我们来看一段入口Activity的注册代码:

<activity android:name=".MainActivity"
           >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

入口Activity和普通Activity唯一不同的地方就是入口Activity多了一个过滤器,对于包含了过滤器的组件,意味着该组件可以提供给外部的其他应用来使用,它的exported属性默认为true,相反,如果一个组件不包含任何过滤器,那么意味着该组件只能通过指定明确的类名来调用,也就是说该组件只能在应用程序的内部使用,在这种情况下,exported属性的默认值是false。

还有一个小技巧:如果你想知道当前手机显示了哪些在运行的任务栈,你可以在Android studio中的Terminal中输入 adb shell dumpsys activity activities。就会出现你想要的信息。我们可以关注 ActivityRecord{2290792d u0 com.hjr.heidoulib/.module.activity.TestComponentNameActivity t9}来获取当前应用所显示的界面的包名跟类名。