转于:http://iteches.com/archives/18829
---------------------------------------------------------
Android – 使用特定的URL开启应用程序
在iPhone、Android或Windows Phone 7的环境都支持透过URL中特定的Schema来启动特定的应用程序,
例如:「zune:\\」、「contnet:\\」…等,该目的告诉系统有一特定的连结要被启动,请系统动找到注册
对应该Schema的程序并且启动它,如果想要夹参数带入应用程序中,写法就如同透过QueryString的方式
将Key/Value带入URL中,由程序启动时根据特定的参数取得值来完成任务。
要完成这样的功能,在开发Android时要修改那些部分与概念,往下来概要的说明:
〉AndroidManifest.xml File:
该份档案描述整个Android应用程序的所有功能、使用的Library、要求手机设备提供的权限控制与存取范围,
因此,今天要增加对特定URL的处理,也正是从该档案下手告诉Android系统该应用程序将对某特定的URL Schema
有所反应,当系统接收到该URL时自动启动应用程序。
A. Intent:
Android应用程序里三个重要组件:activities、services与boardcast services,它们之间传递讯息的对象即是:Intent。
Intent可被不同的应用程序建立当作互相沟通的主要接口,它本身是个对象用于保存讯息交换的内容、Anrdoid固定参等,
在三种不同组件使用方式也有所不同,如下:
‧Activity:
透过Context.startActivity()将Intent带到指定的Activity之中;并且实作Activity.startActivityForResult()接受由另一个
Activity回传的Intent内容(至于回传的内容会透过setResult()的方式将Intent放入);
‧Service:
透过Context.startService()去初始化Service并且将Intent传递给Service让它进行处理;另外可透过Context.bindService()
去选择特定的Service与Intent互相连接;
‧Boardcast Service:
例如透过Context.sendBoardcast()将Intent发送给任何的Boardcast,或是透过Context.sendStickyBroadcast()来送给有兴
趣的Boardcast service。由于Intent里需指定特定的system code不然该Intent就没有被特定的boardcast service查觉到。
了解Intent怎么在三种组件中使用后,接下来针对Intent对象的结构进行说明:
A-1. Component Name:
该属性为ComponentName对象,它代表Intent目标的完整名称,例如:com.example.project.app.FreneticActivity;
例如:由Main的activity移到Second的activity,然而在Second收到Intent它的ComponentName即为完整的Second命
名空间,而Package即与AndroidManifest.xml的相同,但移动到不同的命名空间即不同。如下图:
可透过setComponent()、)" target=_blank>setClass()、setClassName()与getComponent()来操作ComponentName的数据。
A-2. Action:(重要)
Action是Intent中重点部分,它定义该Intent要负责的任务是什么,例如:我想要拨打电话,即会指定该Intent的Action
为ACTION_CALL并在setData指定Uri为tel:0932xxxxxx,启动并直接拨打电话。相对的,如果是broadcast receiver一样透
过Action的指定,让Intent告诉Android系统当收到电量过低时,要通知自己注册的broadcast receiver来完成指定的任务。
Constant Target component Action ACTION_CALL activity 启动一个拨打电话。 ACTION_EDIT activity 显示数据提供用户有编辑能力。 ACTION_MAIN activity 预设第一个启动的activity,没有数据输入和回传数据输出。 ACTION_SYNC activity 设备与伺服务同步数据。 ACTION_VIEW activity 显示数据不支持编辑能力。 ACTION_DIAL activity 启动拨号程序并填入拨号的号码。 ACTION_BATTERY_LOW broadcast receiver 当电量过低时发出的警告。 ACTION_HEADSET_PLUG broadcast receiver 设备已经插入了耳机,或将它拔掉。 ACTION_SCREEN_ON broadcast receiver 屏幕被打开。 ACTION_TIMEZONE_CHANGED broadcast receiver 时区设置发生了变化。
另外,Action也决定的Intent的数据结构,特别是:data与extras。然而,Action除了上述Android API定义的之外,
(随着Android版本不同有更多的Action可用),其实也可以自定义。
A-3. Category:
用于描述该组件(activity, service, broadcast service)要处理的Intent类型,然而,该Category可任意数量的去描述该组件
要处理的Intent对象。如下列的描述类型:
Constant Meaning CATEGORY_BROWSABLE 目标activity可以安全地被透过Browser中显示的link(或uri)来启动,例如:透过image或mail来启动对应的应用程序。 CATEGORY_GADGET activity可以被嵌入至另一个hosts gadgets中的activity。 CATEGORY_HOME activity呈现于HOME screen,也就是当用户按下实体键盘中的HOME时,用户第一个看到的画面。例如:GO桌面即是这种类型的组件。 CATEGORY_LAUNCHER activity是task的初始activity,并且是application launcher的top-level。该类型常设定应用程序第一个启动的activity,并且配合ACTION_MAIN一同定义该activity。 CATEGORY_PREFFRENCE 目标activity是一个preference panel。
A-4. Data:
利用URI为概念的传递格式,定义了URI与MIME的数据类型。不同的Action搭配不同类型Data格式。例如:ACTION_EDIT,
Datal部分需要包括要编辑的文件的URI;ACTION_CALL则Data为要拨打电话的URI:tel:0932xxxxx;如果是ACTION_VIEW,
则Data为http:URI指定的activity收到后将下载或开启对应的程序将数据显示出来。
另外,在找到Intent的对应后,除了取出URI之外,更要看它定义的MIME为何种类型,例如:video/mpeg,则代表要开启
的数据类型为mpeg的影片资料。
通常透过URI也可以推断出相关的应用,例如:Content://即是Android的Content Provider机制,应用程序之间如果有实作
支持Content Provider那么别的应用程序即可以透过Content://来撷取它开放出来的资源。
透过setData()来定义URI,使用setType()定义 MIME类型或是setDataAndType()同时定义二者。
A-5. Extras:
它是一个Key/Value的数据格式,当作Intent要夹带参数的用途。某些Action透过特定的Extras来传递给目标的activity,
做为初始化或识别的任务,例如:ACTION_TIMEZONE_CHANGED有指定的extra key:time-zone或者是ACTION_HEADSET_PLUG
有指定的extra key:state。这样做的目的在于让Intent不一定只能依靠URI传递参数,所以当二个Activity互动时,也可以
透过该方式来进行数据的传递。
A-6. Flags:
Flags通常用于指示Android系统如何执行activity,例如:activity属于那一个Task来执行;或者如何对待启动的activity等,
也可以把一些特殊定义的标签设定于此,让Intent可以夹带特定的识别代码。
介绍了Intent后,其实可以了解Android在三种组件互动或是透过Browser互动,均是透过Intent概念来进行互动。因此,
最后就介绍如果要让自己的应用程序可以接受URL Schema所影响的Intent要设定的参数有那些,进一步去了解Intent-Filter的特性。
B. Intent-filter:
Intent在Anrdoid开发文件里可分成二个部分:
(1) Explicit intents:Intent指定特定目标对象,通常都是内部Project中的activity或是其他组件互相沟通。 。
(2) Implicit intents:Intent不需要特定目标对象,通常用于激活其他应用程序的组件。
Android在使用Explicit intents时,由于目标比较明确比较没有问题,但在遇到Implicit intents时,系统必须寻找到最匹配的组件
来处理Intent,例如:在AndroidManifest.xml定义Activity、Service或Broadcast Receiver来处理特定的Intent(Intent-Filter)。
透过Intent-Filter定义了组件需要处理的explicit、implicit Intents,但如果组件没有定义Intent-Filter的话,该组件只能处理explicit
intents。
然而,每一个Intent-Filter描述组件(activity, service, broadcast receiver)处理的目标与特性,每一个filter是实例化IntentFilter
类别,filter有三个主要的重点:(1). action;(2). data;(3). category;
‧action:
1: <action android:name="string" />
定义该组件要面对的Action对象为何。name属性通常是使用Intent描述的Action属性,自己的应用程序里如果有自定义要处理的
Action也可以透过自定义的字符串常数来说明。在Intent-Filter中至少要有一个<action>标签。
‧category:
1: <intent-filter>
2: <category android:name="android.intent.category.DEFAULT" />
3: <category android:name="android.intent.category.BROWSABLE" />
4: </intent-filter>
上述提到有关intent中的category描述不太一样,在intent中是直接使用上述提到的常数,但在AndroidManifest.xml中定义的
需要使用完整的字符串来加以描述,例如:android.intent.category.BROWSABLE与intent中使用的CATEGORY_BROWSABLE相似。
‧data:
1: <intent-filter>
2: <data android:mimeType="video/mpeg" android:scheme="http" />
3: <data android:mimeType="audio/mpeg" android:scheme="http" />
4: </intent-filter>
每一个<data>定义了处理的URI与data type(MIME media type),属性包括:schema、host、port与path,其格式如下:
URI:{schema://host:port/path},定义Android如收到相同URI的格式需通当定义这个data的组件;
[注意]
‧透过URL启动的与透过手动启动的应用程序二者是不同的,前者是依赖着启动者的线程所启动,后者则是独立的线程,
二者其实控制的功能都是一样的,最大的差异在于透过URL启动的应用程序,在长按「实体home键」要切换响应用程序时,
它不会出现暂存清单中。
〉范例说明:
1. 实作一个网页里面有一个特定的URL,例如:「myapp://Order?SID=201112301333001」;
1: <html xmlns="http://www.w3.org/1999/xhtml" >
2: <head runat="server">
3: <title>使用特定的URL开启应用程序</title>
4: </head>
5: <body>
6: <form id="form1" runat="server">
7: <div>
8: <!-- 定义特定的URL与Schema -->
9: <a href="myapp://Order?SID=201112301333001">您有一封新单据。</a>
10: </div>
11: </form>
12: </body>
13: </html>
2. 建立透过URL启动后要处理的Activity与撷取参数的功能;
在OnCreate事件中建立取得由URL启动应用程序带入的值,这些数据按照Android的作法会放置于Intent的对象中。
1: @Override
2: protected void onCreate(Bundle savedInstanceState) {
3: super.onCreate(savedInstanceState);
4: setContentView(R.layout.urlopen);
5: //取得要显示参数的TextView对象
6: TextView tView = (TextView) findViewById(R.id.textView1);
7:
8: //取得URL所带进来的Intent对象
9: Intent tIntent = this.getIntent();
10: //取得Schema,值为:myapp
11: String tSchema = tIntent.getScheme();
12: //取得URL
13: Uri myURI = tIntent.getData();
14: if (myURI != null) {
15: //取得URL中的Query String参数
16: String tValue = myURI.getQueryParameter("SID");
17: tView.setText(tValue);
18: }
19: }
3. 定义AndroidManifest.xml中支持URL的Schema与互相的相关参数;
1: <application android:icon="@drawable/icon" android:label="@string/app_name">
2: <!-- 主程序-->
3: <activity android:name=".Main"
4: android:label="@string/app_name">
5: <intent-filter>
6: <action android:name="android.intent.action.MAIN" />
7: <category android:name="android.intent.category.LAUNCHER" />
8: </intent-filter>
9: </activity>
10: <!-- 接收由URL启动应用程序后第一个要进入的Activity -->
11: <activity android:name=".Second">
12: <intent-filter>
13: <!-- 宣告该Activity的Action主要类型 -->
14: <action android:name="android.intent.action.VIEW"/>
15: <!-- 定义该Activity支持浏览模式 -->
16: <category android:name="android.intent.category.BROWSABLE"/>
17: <category android:name="android.intent.category.DEFAULT"/>
18: <!-- 定义要处理的URL Schema -->
19: <data android:scheme="myapp"/>
20: </intent-filter>
21: </activity>
22: </application>
======
以上是介绍如何透过特定的URL来启动自己的应用程序,由于该用途很适合结合目前公司既有程序,
因为不想特别公开联机信息出来时,透过URL夹带参数给已安装于手机中的应用程序,再与后端系统
进行互动,既可以做到与设备的结合更可以让部分安全性得到保护。