一般的 Android 应用程序主要由 Activity、BroadcastReceiver、Service、ContentProvider 四大核心组件构成:Activity 作为 Android 应用的表示层通过继承android.app.Activity类实现应用程序每一屏的显示,其是直接和用户交互的窗口。Service 组件主要应用在需要长时间运行于后台且没有用户交互界面或间歇性交互的场合。例如音乐播放器,当退出播放器控制台进行其他操作时,音乐仍旧在后台持续播放。当需要停止播放时,则需要重新进入播放器控制台进行操作。这种间歇性的用户交互就是通过
Service 组件实现的。广播 Broadcast 在 Android 系统中有着非常广泛的使用,而用于接收和处理广播消息的组件就是 BroadcastReceiver。若希望应用能够在系统启动后就立刻自动运行时,该应用就必须要接收和处理 BOOT_COMPLETED 的广播消息。Android 与其他传统手机操作系统不太一样的另一特征是数据文件和数据库数据等数据信息对于 Android 应用来说都是私有的,若应用需要将私有的数据提供给其他应用时就需要使用 Content Provider 组件。应用程序使用该组件提供的一整套标准接口与数据交互而不需要关心数据的具体位置。
这四种核心组件中除了 ContentProvider 需要使用 ContentReceiver 连接之外,其余三大核心组件都需要使用 Intent 消息互通。其支持 Android 设备上安装的任意应用程序组件间的交互,这就把设备从一个相互独立的应用集合转变为一个开放互联的移动平台。可见 Intent 在 Android 应用开发中具有非常重要的地位。
Intent在谷歌的官方文档中是这样定义的,It is a passive data structure holding an abstract description of an operation to be performed.它是一种数据结构,抽象描述一次将要被执行的操作,其作用是在程序运行过程中连接两个不同的组件。
Intent 对象通常使用六个主要组成部分来抽象描述的请求信息,它们是组件名称,行为,数据 / 数据类型,类别,附加信息和标示位:(其实就对应mainfest.xml声明组件的时候定义的各种信息)比如Action,Activity。class,data,type,category等信息。
组建名称:
Intent 对象使用组件名称描述传递消息的目标组件。这种明确指定目标组件名称的 Intent 称之为显式 Intent,系统会将显式 Intent 直接发送给目标组件。由于显式 Intent 需要指定目标组件,所以其更多的使用在应用内部组件间消息的传递。
Action 行为记录了 Intent 消息将要执行的行为字符串或者将要广播的行为字符串。Android 系统虽然已经在 Intent 类中以静态常量的形式预先定义了一系列与系统有关的行为(比如发信息,打电话等的常量),但同样支持使用自定义行为字符串来触发应用中其他的组件(可以结合内容提供者中暴露给外界的主机名的部分来综合考虑)。需要注意的是自定义行为字符串需要使用应用的全包名形式命名。如下列举几个常见的默认的Action:
public static final String ACTION_MAIN = "android.intent.action.MAIN";
public static final String ACTION_VIEW = "android.intent.action.VIEW";
public static final String ACTION_WEB_SEARCH = "android.intent.action.WEB_SEARCH";
public static final String ACTION_CALL = "android.intent.action.CALL";
数据 / 数据类型保存了目标组件将要操作的数据和数据类型,这些信息以 URI 的形式表示存储在 Intent 中。如在上述例子中,ACTION_CALL 行为需要操作的电话号码就是以“tel://number”的形式保存。
类别属性主要描述被请求组件或执行行为动作的额外信息。Android 系统也为类别定义了一系列的静态常量字符串来表示 Intent 不同类别。例如 android.intent.category.LAUNCHER表示目标组件是应用程序中最优先被执行的组件。
为组件提供的扩展信息或额外的数据被保存在 Intent 对象的附加信息中。附加信息采用键值对的结构,以 Bundle 对象的形式保存在 Intent 当中。
标示位主要标示如何触发目标组件以及如何看待被触发的目标组件。例如标示被触发的组件应该属于哪一个任务或者触发的组件是否是最近的 activity 等等。标示位可以是多个标示符的组合。
案例:
下面举两个例子,理解隐式意图和显示意图:
需求:点击按钮,第一个Activity跳转到第二个Activity。
先看隐式意图:
案列一:
public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); } public void click(View view){ /* * 打action * 人 数据 *附件的数据Category类别 */ Intent intent =new Intent(); intent.setAction("com.example.yinshi.open2"); intent.addCategory(Intent.CATEGORY_DEFAULT); //URL:统一资源定位符 http https ftp rtsp: //URI:统一资源标识符,url是uri的一个子集 // intent.setData(Uri.parse("jianren:张三")); //intent.setType("application/person"); //intent.setDataAndType(Uri.parse("jianren:张三"), "application/person"); intent.setDataAndType(Uri.parse("jianren://www.baidu.com"), "application/person"); startActivity(intent); } }
配置文件:
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.yinshi" android:versionCode="1" android:versionName="1.0" > <uses-sdk android:minSdkVersion="14" android:targetSdkVersion="17" /> <application android:allowBackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:theme="@style/AppTheme" > <activity android:name=".MainActivity" android:label="@string/app_name" > <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> <strong> <span style="color:#990000;"> <activity android:name=".SecondActivity"> <intent-filter> <action android:name="com.example.yinshi.open2" /> <category android:name="android.intent.category.DEFAULT"/> <data android:mimeType="application/person"/> <!-- <data android:scheme="jianren"/> --> <data android:scheme="jianren" android:host="www.baidu.com"/> </intent-filter> </activity></span></strong> </application> </manifest>
使用隐式意图的时候,注意运用的时候必须的步骤,在配置文件中需要配置:1)跳转到那个Activity 2)动作action 3)类别category 4)数据及约束
必要的布局文件:
<pre name="code" class="java"><RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.example.yinshi.MainActivity" > <Button android:onClick="click" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerHorizontal="true" android:layout_centerVertical="true" android:text="隐式意图激活界面" /> </RelativeLayout> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.example.yinshi.MainActivity" > <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerHorizontal="true" android:layout_centerVertical="true" android:text="我是第二个界面" /> </RelativeLayout>
案例二:人品计算器 理解显示意图
学习内容:1)显示意图的理解 2)将第一个Activity的内容传递到第二个Activity中去
public class MainActivity extends Activity { private EditText et_name; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); et_name=(EditText) findViewById(R.id.et_name); } public void enter(View vier){ String name=et_name.getText().toString().trim(); if(TextUtils.isEmpty(name)){ Toast.makeText(this, "请输入姓名", 0).show(); } //意图 <strong>Intent intent=new Intent(this,CalcActivity.class);//创建了一个空的意图,两个参数【自己的包空间,第二个Activity的字节码】 intent.putExtra("name", name); //传递数据,类似于Java中的request put内容</strong> //显示意图[把动作写死] //intent.setClassName(getPackageName(), "com.example.wrp.CalcActivity");//不指定动作,也不指定数据,直接指定要激活的组件 //激活意图 <strong>startActivity(intent);</strong> //隐式意图 //intent.setAction(action); //Intent.setData(data); } }
import java.util.Random; import android.app.Activity; import android.content.Intent; import android.os.Bundle; import android.view.Menu; import android.view.MenuItem; import android.widget.TextView; public class CalcActivity extends Activity { private TextView tv_result; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_calc); tv_result=(TextView) findViewById(R.id.tv_result); //得到传递过来的Intent <strong>Intent intent=getIntent(); String name=intent.getStringExtra("name");</strong> //定义了一个字节数组result,这个字节数组是将字符串转换成字节数组存放在result中。 byte[] result=name.getBytes(); int number=0; //将byte数组里面的每个byte转换成十进制数加起来 for(byte b:result){ number+=b&0xff; } //名字可能为汉字,比较长得到的负数 int sorce=Math.abs(number)%100; tv_result.setText(name+"的人品:"+ sorce); } }
需要的布局文件:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.example.wrp.MainActivity" > <Button android:id="@+id/bt_calc" android:onClick="enter" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerHorizontal="true" android:layout_centerVertical="true" android:text="点击人品计算器"/> <ImageView android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="@drawable/rp" android:layout_centerHorizontal="true" /> <EditText android:id="@+id/et_name" android:layout_centerHorizontal="true" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_above="@+id/bt_calc" android:text="请输入要计算的姓名" /> </RelativeLayout> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.example.wrp.CalcActivity" > <TextView android:id="@+id/tv_result" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerHorizontal="true" android:layout_centerVertical="true" android:text="我是计算器页面" /> </RelativeLayout>
案例三:使用隐式意图实现浏览器的访问
学习内容:1)真正实际应用的时候怎么使用隐式意图 2)当单机按钮后,跳转到百度页面。
启发:当我们需要访问激活别人的程序的时候,只需要知道别人提供的动作,数据即可。
Android自带的程序源码,可以参考学习,对于程序的Intent可以参考。Android自带程序源码下载地址:http://download.csdn.net/detail/itjavawfc/8173753
package com.example.dakailiulanqi; import android.app.Activity; import android.content.Intent; import android.net.Uri; import android.os.Bundle; import android.view.Menu; import android.view.MenuItem; import android.view.View; public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); } public void click(View view){ <strong><span style="color:#990000;"> //显示意图:必须制定要激活的组件的完整包名和类名(应用程序间耦合在一起) //一般激活自己应用的组件 //隐式意图,必须制定要动作和数据就可以(应用程序间无耦合) //激活别人写的应用,不需要关心对方的包名和类名</span> Intent intent=new Intent(); //intent.setClassName("com.android.browser", "com.android.browser.BrowserActivity"); intent.setAction("android.intent.action.VIEW"); intent.addCategory("android.intent.category.DEFAULT"); intent.addCategory("android.intent.category.BROWSABLE"); intent.setData(Uri.parse("http://www.baidu.com")); startActivity(intent);</strong> } }
配置文件不需要任何修改,只需要添加访问互联网权限即可。
布局文件:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.example.dakailiulanqi.MainActivity" > <Button android:onClick="click" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerHorizontal="true" android:layout_centerVertical="true" android:text="打开百度" /> </RelativeLayout>
效果: