Intent 的使用——第一行代码读书笔记(3)

Intent 介绍

Intent 是 Android 程序中各组件之间进行交互的一种重要方式,它不仅可以指明当前组件想要执行的动作,还可以在不同组件间传递数据。Intent 一般被用于启动活动、启动服务、或是发送广播等场景。Intent 的用法大致分为两种,显式 intent 和隐式 intent。<br />
既然需要涉及到第二个 Activity,那我们就先来创建一个新的 Activity,命名为 SecondActivity.

  • Step 1 新建 second_layout.xml 文件
<?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:id="@+id/button_2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Button 2"/>
</LinearLayout>
  • Step 2 新建活动 SecondActivity 继承自 Activity
public class SecondActivity extends Activity {
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        setContentView(R.layout.second_layout);
}
}
  • Step 3 在 AndroidManifest.xml 中为 SecondActivity 注册
<activity android:name=".SecondActivity">
</activity>

显式 Intent

Intent 有多个构造函数的重载,其中一个是 Intent(Context packageContext,Class<?>cls),这个构造函数接受两个参数,第一个 Context 是提供一个启动活动的上下文,第二个参数 Class 是指定想要启动的目标活动。通过此函数可以构建出 Intent 的“意图”。Activity 类中提供了一个 startActivity() 方法,此方法用于启动活动,它接收一个 Intent 参数。<br />
修改 FirstActivity 中按钮的点击事件,代码如下:

button1.setOnClickListener(new View.OnClickListener() {
   @Override
   public void onClick(View view) {
      Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
      startActivity(intent);
   }
});

使用这种方式来启动活动,Intent 的意图很明显,所以称为显式 Intent。

隐式 Intent

并不明确指出我们想要启动哪一个活动,而是指定了一系列更抽象的 action 和 category 等,然后又系统去分析这个 Intent 并帮我们找到合适的活动去启动。

  • 配置 AndroidManifest.xml 中 <activity> 标签下 <intent-filter> 中的 <action> 和 <category> 可以指定当前活动能够相应的 action 和 category。
<activity android:name=".SecondActivity">
   <intent-filter>
       <action android:name="com.example.activitytest.ACTION_START"/>
       <category android:name="android.intent.category.DEFAULT"/>
   </intent-filter>
</activity>
  • 修改 FirstActivity 中按钮的点击事件,使 SecondActivity 能够响应。
button1.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
        Intent intent = new Intent("com.example.activitytest.ACTION_START");
        startActivity(intent);
    }
});

为什么这里没有指定 category 呢?因为 android.intent.category.DEFAULT 是一种默认的 category,startActivity() 方法会自动把 category 添加到 Intent 中。<br />
每个 Intent 中只能指定一个 action,但却能够指定多个 category。

  • 再为 SecondActivity 添加一个 category。
<activity android:name=".SecondActivity">
     <intent-filter>
         <action android:name="com.example.activitytest.ACTION_START"/>
         <category android:name="android.intent.category.DEFAULT"/>
         <category android:name="com.example,activitytest.MY_CATEGORY"/>
     </intent-filter>
</activity>
  • 再次修改 FirstActivity 中按钮的点击事件,使 SecondActivity 能够响应。
button1.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
        Intent intent = new Intent("com.example.activitytest.ACTION_START");
        intent.addCategory("com.example,activitytest.MY_CATEGORY");
        startActivity(intent);
    }
});

调用 Intent 中的 addCategory() 方法来添加一个 category,这样 SecondActivity 就能够正常响应了。

更多的隐式 Intent 用法

  • 修改 FirstActivity 中按钮的点击事件,使之能够通过系统自带的浏览器打开网页。
button1.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
        Intent intent = new Intent(Intent.ACTION_VIEW);
        intent.setData(Uri.parse("http://www.baidu.com"));
        startActivity(intent);
    }
});

Intent 的 action 是一个 Android 系统内置的动作,常量值为 android.intent.action.VIEW。然后通过 Uri.parse() 方法,将一个网址字符串解析成一个 Uri 对象,再调用 Intent 的 setData() 方法把 Uri 对象传递进去。setDate() 方法接收一个 Uri 对象,主要用于指定当前 Intent 正在操作的数据,数据通常是以字符串形式传递到 Uri.parse() 方法中解析产生。

<data> 标签的使用

在 <intent-filter> 标签中再配置一个 <data> 标签,可用于更精准的指定当前活动能够响应什么类型的数据。<data> 标签主要可以配置以下内容:

  1. android:scheme <br />
    用于指定数据协议部分
  2. android:host <br />
    用于指定数据主机名部分
  3. android:port <br />
    用于指定数据端口部分
  4. android:path <br />
    用于指定主机名和端口之后部分
  5. android:mimeType <br />
    用于指定可以处理的数据类型

只有 <data> 标签中指定的内容和 Intent 中携带的 Data 完全一致时,当前活动才能够响应该 Intent。例如:

  • 新建 third_layout.xml 布局文件
<?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:id="@+id/button_3"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" 
        android:text="Button 3"/>
</LinearLayout>
  • 新建活动 ThirdActivity 继承自 Activity
public class ThirdActivity extends Activity {
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        setContentView(R.layout.third_layout);
    }
}
  • 在 AndroidManifest.xml 中为 ThirdActivity 注册
<activity android:name=".ThirdActivity">
    <intent-filter>
        <action android:name="android.intent.action.VIEW"/>
        <category android:name="android.intent.category.DEFAULT"/>
        <data android:scheme="http"/>
    </intent-filter>
</activity>

此时运行点击 FirstActivity 的按钮会弹出一个列表,让你选择是用自带浏览器打开还是使用新建的 ThirdActivity 打开。

调用系统拨号界面

button1.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
        Intent intent = new Intent(Intent.ACTION_DIAL);
        intent.setData(Uri.parse("tel:10086"));
        startActivity(intent);
    }
});

向下一个活动传递数据

启动活动时传递数据思路很简单,Intent 中提供了一系列的 putExtra() 方法的重载,把想要传递的数据暂存在 Intent 中,启动另一个活动后,再把数据从 Intent 中取出来。

button1.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
        String data = "Hello SecondActivity";       
        Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
        intent.putExtra("extra_data",data);
        startActivity(intent);
    }
});

putExtra() 方法接收两个参数,第一个是键,第二个是值。

 protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    requestWindowFeature(Window.FEATURE_NO_TITLE);
    setContentView(R.layout.second_layout);
    Intent intent = getIntent();
    String data = intent.getStringExtra("extra_data");
    Log.d("SecondActivity",data);
}

在 SecondActivity 中将传递的数据取出,并打印出来。通过 getIntent() 方法获取用于启动 SecondActivity 的 Intent ,然后调用 getStringExtra() 方法取出数据。

返回数据给上一活动

Activity 中有一个 startActivityForResult() 方法也用于启动活动,但这方法期望在活动销毁时返回一个结果给上一个活动。

button1.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
        Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
        startActivityForResult(intent,1);
    }
});

使用 startActivityForResult() 方法来启动 SecondActivity ,startActivityForResult() 方法接收两个参数,第一个参数是 Intent,第二个参数是请求码,用于在回调中判断数据来源。

  • 为 SecondActivity 按钮注册点击事件,添加返回数据逻辑
button2.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
        Intent intent = new Intent();
        intent.putExtra("data_return","Hello FirstActivity");
        setResult(RESULT_OK,intent);
        finish();
    }
});

setResult() 方法专门用于向上一个活动返回数据,接收两个参数,第一个是向上一个活动返回处理结果,第二个参数是把带有数据的 Intent 传递回去,然后调用 finish() 方法销毁当前活动。<br />
使用 startActivityForResult() 方法来启动 SecondActivity,在 SecondActivity 销毁之后会回调上一个活动的 onActivityResult() 方法,所以需在 FristActivity 上重写这个方法来获取数据。

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    switch (requestCode){
        case 1:
            if (resultCode == RESULT_OK){
                String returnedData = data.getStringExtra("data_return");
                Log.d("FirstActivity",returnedData);
            }
            break;
        default:
    }
}

onActivityResult() 方法带有三个参数,第一个是启动活动时传入的请求码,第二个是返回时的处理结果,第三个是携带返回数据的 Intent。<br /><br />
如果用户点击的是 Back 键,而不是点击按钮怎么办?重写 onBackPressed() 方法。

@Override
public void onBackPressed() {
    Intent intent = new Intent();
    intent.putExtra("data_return","Hello FirstActivity");
    setResult(RESULT_OK,intent);
    finish();
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 157,298评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,701评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 107,078评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,687评论 0 202
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,018评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,410评论 1 211
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,729评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,412评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,124评论 1 239
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,379评论 2 242
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,903评论 1 257
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,268评论 2 251
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,894评论 3 233
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,014评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,770评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,435评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,312评论 2 260

推荐阅读更多精彩内容