Action、Category属性与intent-filter配置

时间:2022-08-21 16:26:41

Intent的Action、Category属性都是一个普通的字符串,其中Action代表该Intent所要完成的一个抽象“动作“,而Category则用于为Action增加额外的附加类别信息。通常Action属性会与Category属性结合使用。

Action要完成的只是一个抽象的动作,这个动作具体由哪个组件(或许是Activity,或者是BroadcastReceiver)来完成,Action这个字符串本身并不管。比如Android提供的标准Action:Intent.ACTION _VIEW,它指标是一个抽象的查看操作,但具体查看什么、启动哪个Activity来查看,Intent. ACTION _VIEW并不知道——这取决于Activity的< intent-filter …/ >配置,只要某个Activity的< intent-filter …/ >配置中包含了该ACTION _VIEW,该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
android:name=".SecondActivity"
android:label="@string/title_activity_second"
android:theme="@style/AppTheme.NoActionBar">

<intent-filter>
<action android:name="com.mystudy.kibi.action.TEST_ACTION_ONE" />
</intent-filter>
</activity>
public class MainActivity extends AppCompatActivity {
final String ACTION = "com.mystudy.kibi.action.TEST_ACTION_ONE";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Intent intent = new Intent();
intent.setAction(ACTION);
startActivity(intent);
}
}

上面的程序会成功实现跳转吗?
答案是:报错,无法跳转,原因分析如下:


页面跳转取决于Activity配置中< intent-filter …/ >元素的配置。
< intent-filter …/ >元素里通常包含如下子元素:

  • 0~N个< action …/ >子元素
  • 0~N个< category …/ >子元素
  • 0~1个< data …/ >子元素
    (< intent-filter …/ >元素也可是< service …/ >、< receiver …/ >两个元素的子元素,用于表明他们可以响应的Intent。)

由于上面的程序制定启动Action属性为MainActivity.ACTION常量(常量值为:com.mystudy.kibi.action.TEST_ACTION_ONE)的Activity,也就要求启动Activity对于的配置元素的< intent-filter …/ >元素里只好包含一个如下的< action …/ >子元素:

<action android:name="com.mystudy.kibi.action.TEST_ACTION_ONE" />

要注意的是:

  • 一个Intent对象最多只能包含一个Action属性,程序可调用Intent的setAction(String str)方法来设置Action属性值
  • 一个Intent对象可以包含多个Category属性,程序可调用Intent的addCategory(String str)方法来为Category属性
  • 当程序创建Intent时,该Intent默认启动Category属性值为Intent.CATEGORY_DEFAULT常量(常量值为android.intent.category.DEFAULT)的组件

因此,虽然上面的程序并未指定目标的Intent的Category属性,但该Intent已有一个值为android.intent.category.DEFAULT的Intent.CATEGORY_DEFAULT常量,因此被启动Activity对应的配置元素的< intent-filter …/ >元素里至少还要包含一个如下的< category …/ >子元素:

<category android:name="android.intent.category.DEFAULT" />

整体测试代码如下所示:

<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name=".SecondActivity"
android:label="@string/title_activity_second"
android:theme="@style/AppTheme.NoActionBar">

<intent-filter>
<action android:name="com.mystudy.kibi.action.TEST_ACTION_ONE" />
<action android:name="com.mystudy.kibi.action.TEST_ACTION_TWO" />
<action android:name="com.mystudy.kibi.action.TEST_ACTION_THREE" />
<category android:name="android.intent.category.DEFAULT" />
<category android:name="com.mystudy.kibi.category.TEST_CATEGORY_ONE" />
<category android:name="com.mystudy.kibi.category.TEST_CATEGORY_TWO" />
<category android:name="com.mystudy.kibi.category.TEST_CATEGORY_THREE" />
</intent-filter>
</activity>
<activity android:name=".ThreeActivity">
<intent-filter>
<action android:name="com.mystudy.kibi.action.TEST_ACTION_ONE" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</activity>
public class MainActivity extends AppCompatActivity {
final String ACTION = "com.mystudy.kibi.action.TEST_ACTION_ONE";
final String Category = "com.mystudy.kibi.category.TEST_CATEGORY_ONE";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Intent intent = new Intent();
intent.setAction(ACTION);
intent.addCategory(Category);
startActivity(intent);
}
}
public class SecondActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_second);
Intent intent = getIntent();
Log.e("action",intent.getAction());
Log.e("category",intent.getCategories()+"");
}
}

测试结果:

06-29 22:44:06.014 23818-23818/com.mystudy.kibi.networktype E/action: com.mystudy.kibi.action.TEST_ACTION_ONE
06-29 22:44:06.014 23818-23818/com.mystudy.kibi.networktype E/category: {com.mystudy.kibi.category.TEST_CATEGORY_ONE}