Android学习笔记——APP页面转换与Intent学习

使用Intent保证多活动之间的切换,实现APP页面转换。
同时学习Intent其他诸多功能。

认知Intent

Intent是Android程序中各组件之间进行交互的一种重要方式,不仅指明当前组件想执行的动作,还可以在不同组件之间传递数据。
Intent一般可被用于启动活动、启动服务以及发送广播等场景。初学者可以从启动活动着手认识。
Intent大致分为两类:显示Intent和隐式Intent

显示Intent实现活动转换

  1. 首先创建一个新的活动并自动生成layout文件。
  2. 编辑second_layout.xml和SecondActivity.java,添加按钮2。
  3. 确认Android Studio已经自动为活动完成注册。AndroidManifest.xml。
  4. 重写Button1方法。
    首先构建出一个Intent,传入FirstActivity.this作为上下文,传入 SecondActivity.class 作为目标活动。实现在FirstActivity活动的基础上打开SecondActivity活动。“意图”明显,所以称为显示Intent。
button1.setOnClickListener(new View.OnClickListener(){
	@Override
	public void onClick(View v){
	    Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
	    startActivity(intent);
	//Toast.makeText(FirstActivity.this,"You cicked Button 1",
	//Toast.LENGTH_SHORT).show();
	}
});

分析一下:

  • Intent(Context packageContext, Class<?> cls):这是Intent多个构造函数的重载中的一个
    其中参数一Context要求提供一个启动活动的上下文,参数二Class指定想要启动的目标活动。
  • startActivity()方法:Activity类中一个方法,专用于启动活动的,接收一个Intent参数。

此时点击按钮1就会自动启动SecondActivity活动,按钮2就会被展示出来。
点击Back键就会回到FirstActivity。

隐示Intent实现活动转换

相比于显示Intent,隐式Intent更含蓄。不明确指明要启动哪一个活动,而是指定一系列更为抽象的action和category信息,然后由系统去分析这个Intent,并帮我们找出合适的活动去启动。

所谓合适的活动(挖坑)


打开AndroidManifest.xml文件,在SecondActivity的<activity>标签中添加:

<activity android:name=".SecondActivity">
    <intent-filter>
        <action android:name="com.example.activitytest.ACTION_START"/>
        <category android:name="android.intent.category.DEFAULT"/>
    </intent-filter>
</activity>

分析:
<action>标签中指明当前活动可以响应com.example.activitytest.ACTION_START这个action,而<category>标签包含了一些附加信息,更精确指明了当前的活动能够响应的Intent中还可能带有category。只有<action><category>中的内容同时能够匹配上Intent中指定的action和category时,这个活动才能响应该Intent。


打开FirstActivity.java,修改按钮点击事件:

button1.setOnClickListener(new View.OnClickListener(){
	@Override
	public void onClick(View v){
	    Intent intent = new Intent("com.example.activitytest.ACTION_START");
	    startActivity(intent);
	}
});

分析:
可以看到我们是用Intent的另一个构造函数,直接将action的字符串传了进去,表明我们想要启动能够响应com.example.activitytest.ACTION_START这个action的活动。没有指定<category>是因为上次修改中,标签内容android.intent.category.DEFAULT是一种默认的category,调用startActivity()方法时候会自动将这个category添加到Intent中。
(我就是个无情的搬运工o(╥﹏╥)o似懂非懂)

此时重新运行程序,在FirstActivity界面点一下按钮就会启动SecondActivity。证明<activity>标签下配置的action和category的内容生效了。


每个Intent中只能指定一个action,能指定多个category。尝试添加一个category。
在FirstActivity.java中修改:
调用Intent中的addCategory()方法来添加一个category。这里添加了一个自定义的category。

button1.setOnClickListener(new View.OnClickListener(){
    @Override
    public void onClick(View v){
        Intent intent = new Intent("com.example.activitytest.ACTION_START");
        intent.addCategory("com.example.activitytest.MY_CATEGORY");
        startActivity(intent);
    }
});

在AndroidManifest.xml中修改:
添加自定义的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>

更多隐式Intent的用法

1. 启动其他程序的活动

例如在应用程序中展示一个网页,没必要自己去实现一个浏览器,只需调用系统的浏览器打开这个网页就行了。try:
修改FirstActivity中按钮响应事件的代码:

button1.setOnClickListener(new View.OnClickListener(){
    @Override
    public void onClick(View v){
        Intent intent = new Intent(Intent.ACTION_VIEW);
        intent.setData(Uri.parse("http://www.baidu.com"));
        startActivity(intent);
    }
});

分析一下:

  • Intent.ACTION_VIEW:指定的intent的action。这是一个Android系统内置的动作,常量为android.intent.action.VIEW
  • Uri.parse()方法:将一个网址字符串解析成一个Uri对象。
  • setData()方法:Intent的类方法。接收一个Uri对象,主要用于指定当前Intent正在操作的数据,这些数据通常都是以字符串的形式传入Uri.parse()方法中解析产生的。

与此同时,我们还可以在<intent-filter>标签中再配置一个<data>标签,用于更精准地指定当前活动能够响应什么类型的数据。<data>标签中主要可以配置以下内容,作为规范Intent中携带的Data:

  • android:scheme:指定数据的协议。
  • android:host:指定数据的主机名部分。
  • android:port:指定数据的端口部分。
  • android:path:指定主机名和端口之后的部分,例如一段网址中跟在域名之后的内容。
  • android:mimeType:指定可以处理的数据类型,允许使用通配符方式进行制定。
    只有<data>标签中指定的内容和Intent中携带的Data完全一致时,当前活动才能够响应该Intent。
    此时点击按钮就出现两种选择,一个是直接选择浏览器,另一个ActivityTest则是ThirdActivity活动。也就是系统会选择合适的可能的活动来供用户选择执行。
    在这里插入图片描述

除了http协议,还可以:geo表示显示地理位置,tel表示拨打电话。
如下更改可以直接跳转拨号界面:

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

2.向下一个活动传递数据

Intent中提供了一系列putExtra()方法的重载,可以把我们想要传递的数据暂存在Intent中,启动另一个活动后,再从Intent取出就可以了。
例:在FirstActivity传入一个字符串,在SecondActivity中打印出来。
对FirstActivity修改:

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

分析:本次使用显示Intent。putExtra()方法接收两个参数,第一是键,用于后面从Intent取值;第二是要传递的数据。


对SecondActivity修改:

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

分析:通过getIntent()方法获取到用于启动SecondActivity的Intent,然后调用getStringExtra()方法,传入相应的键值。

  • 字符串数据: getStringExtra()
  • 整型数据:getIntExtra()
  • 布尔型数据:getBooleanExtra()
  • 以此类推

最终通过Logcat可以查看到打印出来的字符串。

3.返回数据给上一个活动

startActivityForResult()方法:用于启动活动,并期望在销毁的时候能够返回一个结果给上一个活动。方法接收两个参数。第一个参数还是Intent;第二个是请求码,用于在之后的回调中判断数据的来源。
例子:
修改FirstActivity:

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

只是将开启活动换成了上述方法。请求码只要是一个唯一值就可以。


修改SecondActivity:

Button button2 = (Button) findViewById(R.id.button_2);
button2.setOnClickListener(new View.OnClickListener(){
    @Override
    public void onClick(View v){
        Intent intent = new Intent();
        intent.putExtra("data_return", "Hello FirstActivity");
        setResult(RESULT_OK, intent);
        finish();
    }
});

分析:给按钮注册点击事件,并添加了返回数据的逻辑。

  • setResult()方法:接收两个参数,第一个用于向上一个活动返回数据,一般只用RESULT_OK或RESULT_CANCELED这两个值;第二个参数则把带有数据的Intent传递回去。

重写onActivityResult()方法用于得到返回的数据:

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

onActivityResult()方法有三个参数:第一个参数requestCode即启动活动时传入的请求码;第二个参数resultCode,即返回数据时传入的处理结果;第三个参数data,携带返回数据的Intent。

由于一个活动中可能调用startActivityForResult()方法启动很多不同的活动,所以要先根据resultCode判断数据来源;确定是需要的数据后根据resultCode判断处理结果是否成功。最后获取Intent数据进行逻辑处理。


此时遇到一个问题:当用户没有点击Button2而是通过Back键返回的,就没办法返回数据了。
解决:
在SecondActivity重写onBackPressed()方法。

@Override
public void onBackPressed() {
    Intent intent = new Intent();
    intent.putExtra("data_return", "Hello FirstActivity");
    setResult(RESULT_OK, intent);
    finish();
}

所执行的操作与点击Button2是基本一致的。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值