组件通信

Author Avatar
Euan 10月 24, 2019
  • 在其它设备中阅读本文章

组件通信

实验目的

1.了解Android的意图方法Intent
2.了解Android活动程序Activity和意图方法Intent之间的关系

实验内容

1.编写程序测试Android活动程序Activity如何进行显示窗体的切换
2.使用Intent在Activity之间传递数据

实验原理

一般情况Android应用程序是由以下四种组件构造而成的:

● 活动
● 广播接收器
● 服务
● 内容提供器

需要注意的是,并不是每个Android应用程序都必须构建这四个组件,有些可能由这些组件的组合而成。
活动是最基本的Android应用程序组件,应用程序中,一个活动通常就是一个单独的屏幕。每一个活动都被实现为一个独立的类,并且从活动基类中继承而来,活动类将会显示由视图控件组成的用户接口,并对事件做出响应。大多数的应用是由多屏幕显示组成。
在这些组件之间的通讯中,主要是由Intent协助完成的。Intent负责对应用中一次操作的动作、动作涉及数据、附加数据进行描述,Android则根据此Intent的描述,负责找到对应的组件,将 Intent传递给调用的组件,并完成组件的调用。Intent不仅可用于应用程序之间,也可用于应用程序内部的Activity/Service之间的交互。因此,Intent在这里起着一个媒体中介的作用,专门提供组件互相调用的相关信息,实现调用者与被调用者之间的解耦。在SDK中给出了Intent作用的表现形式为:
●通过Context.startActivity() 或者Activity.startActivityForResult() 启动一个Activity;
●通过 Context.startService() 启动一个服务,或者通过Context.bindService() 和后台服务交互;
●通过广播方法(如Context.sendBroadcast(),Context.sendOrderedBroadcast()以及Context.sendStickyBroadcast())发给broadcast receivers。

Intent属性的设置,包括以下几点:(以下为XML中定义,当然也可以通过Intent类的方法来获取和设置)
(1)Action,也就是要执行的动作。
SDK中定义了一些标准的动作,如表3-1所示。当然,也可以自定义动作(自定义的动作在使用时,需要加上包名作为前缀,如”com.example.project.SHOW_COLOR”),并可定义相应的Activity来处理我们的自定义动作。
(2)Data,也就是执行动作要操作的数据
Android中采用指向数据的一个URI来表示,如在联系人应用中,一个指向某联系人的URI可能为:content://contacts/1。对于不同的动作,其URI数据的类型是不同的(可以设置type属性指定特定类型数据),如ACTION_EDIT指定Data为文件URI,打电话为tel:URI,访问网络为http:URI,而由content provider提供的数据则为content: URIs。

表3-1 SDK中定义的标准动作
Constant|Target component|Action
:–:|:–:|:–:
ACTION_CALL|activity|Initiate a phone call.
ACTION_EDIT|activity|Display data for the user to edit.
ACTION_MAIN|activity|Start up as the initial activity of a task, with no data input and no returned output.
ACTION_SYNC|activity|Synchronize data on a server with data on the mobile device.
ACTION_BATTERY_LOW|broadcast receiver|A warning that the battery is low.
ACTION_HEADSET_PLUG|broadcast receiver|A headset has been plugged into the device, or unplugged from it.
ACTION_SCREEN_ON|broadcast receiver|The screen has been turned on.
ACTION_TIMEZONE_CHANGED|broadcast receiver|The setting for the time zone has changed.
(3)type(数据类型),显式指定Intent的数据类型(MIME)。一般Intent的数据类型能够根据数据本身进行判定,但是通过设置这个属性,可以强制采用显式指定的类型而不再进行推导。
(4)category(类别),被执行动作的附加信息。例如 LAUNCHER_CATEGORY 表示Intent 的接受者应该在Launcher中作为顶级应用出现;而ALTERNATIVE_CATEGORY表示当前的Intent是一系列的可选动作中的一个,这些动作可以在同一块数据上执行。还有其他的行为如表3-2所示。
表3-2 被执行动作的附加信息
Constant|Meaning
:–:|:–:
CATEGORY_BROWSABLE|The target activity can be safely invoked by the browser to display data referenced by a link — for example, an image or an e-mail message.
CATEGORY_GADGET|The activity can be embedded inside of another activity that hosts gadgets.
CATEGORY_HOME|The activity displays the home screen, the first screen the user sees when the device is turned on or when the HOME key is pressed.
CATEGORY_LAUNCHER|The activity can be the initial activity of a task and is listed in the top-level application launcher.
CATEGORY_PREFERENCE|The target activity is a preference panel.
(5)component(组件),指定Intent的的目标组件的类名称。通常 Android会根据Intent 中包含的其它属性的信息,比如action、data/type、category进行查找,最终找到一个与之匹配的目标组件。但是,如果 component这个属性有指定的话,将直接使用它指定的组件,而不再执行上述查找过程。指定了这个属性以后,Intent的其它所有属性都是可选的。
(6)extras(附加信息),是其它所有附加信息的集合。使用extras可以为组件提供扩展信息,比如,如果要执行“发送电子邮件”这个动作,可以将电子邮件的的标题、正文保存在extras里,传给电子邮件发送组件。

(7)flags(标记),指示Android如何启动目标Activity,设置方法为调用Intent的setFlags方法。常用的Flags参数有:

FLAG_ACTIVITY_CLEAR_TOP
FLAG_ACTIVITY_NEW_TASK
FLAG_ACTIVITY_NO_HISTORY
FLAG_ACTIVITY_SINGLE_TOP

(a)Intent-Filter的定义中一些属性设置的例子


<data android:mimeType=”video/mpeg” android:scheme=”http” . . . />


(b)Intent用法实例
①无参数Activity跳转
Intent it = new Intent(Activity.Main.this, Activity2.class);
startActivity(it);
②向下一个Activity传递数据(使用Bundle和Intent.putExtras)
Intent it = new Intent(Activity.Main.this, Activity2.class);
Bundle bundle=new Bundle();
bundle.putString(“name”, “This is from MainActivity!”);
it.putExtras(bundle); // it.putExtra(“test”, “shuju”);
startActivity(it); // startActivityForResult(it,REQUEST_CODE);
对于数据的获取可以采用:
Bundle bundle=getIntent().getExtras();
String name=bundle.getString(“name”);
(c)向上一个Activity返回结果(使用setResult,针对startActivityForResult(it,REQUEST_CODE)启动的Activity)
Intent intent=getIntent();
Bundle bundle2=new Bundle();
bundle2.putString(“name”, “This is from ShowMsg!”);
intent.putExtras(bundle2);
setResult(RESULT_OK, intent);
(d)回调上一个Activity的结果处理函数(onActivityResult)
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
// TODO Auto-generated method stub
super.onActivityResult(requestCode, resultCode, data);
if (requestCode==REQUEST_CODE){
if(resultCode==RESULT_CANCELED)
setTitle(“cancle”);
else if (resultCode==RESULT_OK) {
String temp=null;
Bundle bundle=data.getExtras();
if(bundle!=null) temp=bundle.getString(“name”);
setTitle(temp);
}
}
}
其他的一些应用实例:
1)打电话
a.叫出拨号程序
Uri uri = Uri.parse(“tel:0800000123”);
Intent it = new Intent(Intent.ACTION_DIAL, uri);
startActivity(it);
b.直接打电话出去
Uri uri = Uri.parse(“tel:0800000123”);
Intent it = new Intent(Intent.ACTION_CALL, uri);
startActivity(it);
使用这些程序,需要在 AndroidManifest.xml 中,加上
传送SMS/MMS
a.调用短信程序
Intent it = new Intent(Intent.ACTION_VIEW, uri);
it.putExtra(“sms_body”, “The SMS text”);
it.setType(“vnd.android-dir/mms-sms”);
startActivity(it);
b.传送消息
Uri uri = Uri.parse(“smsto://0800000123”);
Intent it = new Intent(Intent.ACTION_SENDTO, uri);
it.putExtra(“sms_body”, “The SMS text”);
startActivity(it);
c.传送 MMS
Uri uri = Uri.parse(“content://media/external/images/media/23”);
Intent it = new Intent(Intent.ACTION_SEND);
it.putExtra(“sms_body”, “some text”);
it.putExtra(Intent.EXTRA_STREAM, uri);
it.setType(“image/png”);
startActivity(it);
实验步骤
新建一个IntentDemo应用程序来演示如何利用Intent类来进行显示窗体间的切换。显示的第一个窗体如图3-1所示。

K5nXlQ.png

图3-1启动程序后所显示的第一个窗体
提示:需要在程序中添加一个新的Activity,可以在项目上右键,选择”New”->”Other”,然后选择Android文件夹下的“Android Object”,单击”Next”,选择”Empty Activity”

K5nvOs.png

K5nzmn.png

然后可以设置Activity的属性,单击”Finish”就可以了。系统会自动在AndroidManifest.xml中注册该Activity。

K5njyj.png

点击按钮,启动ActivityNew,此时,通过以下代码向ActivityNew传递数据:
//生成一个Intent对象
Intent intent = new Intent();
//在Intent对象当中添加一个键值对
intent.putExtra(“testIntent”, “123”);
//设置Intent对象要启动的Activity
intent.setClass(Activity01.this, ActivityNew.class);
//通过Intent对象启动另外一个Activity
Activity01.this.startActivity(intent);
ActivityNew启动之后,如图3-2所示,包含一个TextView,用来显示从Activity01传递过来的数据。

K5nOSg.png

图3-2 从Activity01传递过来的数据
点击“发短信”按钮启动发送短信的Activity,如图3-3所示。。
Uri uri = Uri.parse(“smsto: 13912345678”);
Intent intent = new Intent(Intent.ACTION_SENDTO, uri);
intent.putExtra(“sms_body”, “The SMS text”);
startActivity(intent);

提示:按钮需要添加监听器,需要用到前面的知识。类似下面的代码
Button button = (Button)findViewById(R.id.btn);
button.setOnClickListener(new OnClickListener(){
public void onClick(View view){
//todo:添加你需要的代码
}
});

K5uSwq.png

图3-3 发短信的Activity界面
点击发送按钮,系统将会发出短信,如图3-4所示。

K5upT0.png

图3-4 短信发出后的界面

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
ActivityNew
package com.example.thinkpad.intentdemo;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;

public class ActivityNew extends Activity {

@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_new);
// 获取应用程序中的bn按钮
Button bn = (Button) findViewById(R.id.bn);
// 为bn按钮绑定事件监听器
bn.setOnClickListener(new View.OnClickListener()
{
@Override
public void onClick(View source)
{
// 创建需要启动的Activity对应的Intent
Intent intent = new Intent(ActivityNew.this,
SecondActivity.class);
// 启动intent对应的Activity
startActivity(intent);
}
});
}
}

SecondActivity

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
package com.example.thinkpad.intentdemo;

import android.app.Activity;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.telephony.SmsManager;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;

public class SecondActivity extends Activity {
EditText address;
EditText content;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_second);
// 获取页面中收件人地址、短信内容
address = (EditText)findViewById(R.id.address);
content = (EditText)findViewById(R.id.content);
Button bn = (Button)findViewById(R.id.send);
// 使用外部类的实例作为事件监听器
bn.setOnLongClickListener(new org.crazyit.event.third(
this , address, content));
//发短信
Button button2= (Button) findViewById(R.id.send);
button2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String addressStr = address.getText().toString();
String contentStr = content.getText().toString();
Intent intent = new Intent();
intent.setAction(Intent.ACTION_SENDTO);
intent.setData(Uri.parse("smsto:"+addressStr));
intent.putExtra("sms_body",contentStr);
startActivity(intent);
}
});
}
}

third

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
package org.crazyit.event;

import android.app.Activity;
import android.app.PendingIntent;
import android.content.Intent;
import android.telephony.SmsManager;
import android.view.View;
import android.view.View.OnLongClickListener;
import android.widget.EditText;
import android.widget.Toast;

public class third implements OnLongClickListener
{
private Activity act;
private EditText address;
private EditText content;
public third(Activity act, EditText address
, EditText content) {
this.act = act;
this.address = address;
this.content = content;
}
@Override
public boolean onLongClick(View source)
{
String addressStr = address.getText().toString();
String contentStr = content.getText().toString();
// 获取短信管理器
SmsManager smsManager = SmsManager.getDefault();
// 创建发送短信的PendingIntent
PendingIntent sentIntent = PendingIntent.getBroadcast(act
, 0, new Intent(), 0);
// 发送文本短信
smsManager.sendTextMessage(addressStr, null, contentStr
, sentIntent, null);
Toast.makeText(act, "短信发送完成", Toast.LENGTH_LONG).show();
return false;
}
}

layout
activity_new.xml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
<?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/bn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="启动第二个Activity"
/>
</LinearLayout>
activity_second.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/bn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="启动第二个Activity"
/>
</LinearLayout>

K5uCkV.png

K5uPYT.png

K5uifU.png

K5ukpF.png

点击启动第二个Activity启动SecongActivity,第一行填写短信号码,第二行填写短信内容,点击发送后,跳转到系统短信界面,并且自动填写之前输入的短信号码和短信内容

本文使用 CC BY-NC-SA 3.0 中国大陆 协议许可
具体请参见 知识共享协议

本文链接:https://zyhang8.github.io/2019/10/24/android-exp3/