這篇文章主要介紹“Android中的Intent機(jī)制有什么用”,在日常操作中,相信很多人在Android中的Intent機(jī)制有什么用問(wèn)題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”Android中的Intent機(jī)制有什么用”的疑惑有所幫助!接下來(lái),請(qǐng)跟著小編一起來(lái)學(xué)習(xí)吧!
成都創(chuàng)新互聯(lián)公司堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站制作、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時(shí)代的湖里網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!
對(duì)于大型軟件開(kāi)發(fā)經(jīng)驗(yàn)較少的程序員來(lái)說(shuō),這可能是一個(gè)不太容易理解的抽象概念,因?yàn)樗c我們平常使用的簡(jiǎn)單函數(shù)調(diào)用,或者通過(guò)庫(kù)調(diào)用接口的方式不太一樣。 在 Intent 的使用中你看不到直接的函數(shù)調(diào)用,相對(duì)函數(shù)調(diào)用來(lái)說(shuō),Intent 是更為抽象的概念,利用 Intent 所實(shí)現(xiàn)的軟件復(fù)用的粒度是Activity/Service ,比函數(shù)復(fù)用更高一些,另外耦合也更為松散。
Android 中與Intent 相關(guān)的還有 Action/Category 及 Intent Filter 等,另外還有用于廣播的 Intent ,這些元素?fù)诫s在一起,導(dǎo)致初學(xué)者不太容易迅速掌握 Intent 的用法。在講解這些名詞之前,我們先來(lái)從下面的例子中感受一下 Intent 的一些基本用法,看看它能做些什么,之后再來(lái)思考這種機(jī)制背后的意義。
理解 Intent 的關(guān)鍵之一是理解清楚Intent 的兩種基本用法:一種是顯式的 Intent ,即在構(gòu)造 Intent 對(duì)象時(shí)就指定接收者,這種方式與普通的函數(shù)調(diào)用類(lèi)似, 只是復(fù)用的粒度有所差別;另一種是隱式的 Intent ,即Intent 的發(fā)送者在構(gòu)造 Intent 對(duì)象時(shí),并不知道也不關(guān)心接收者是誰(shuí),這種方式與函數(shù)調(diào)用差別比較大,有利于降低發(fā)送者和接收 者之間的耦合。另外 Intent 除了發(fā)送外,還可用于廣播。
下面的一小節(jié)我們來(lái)看看顯式 Intent 的用法。
顯式的Intent(Explicit Intent)
同一個(gè)應(yīng)用程序中的Activity切換
通常一個(gè)應(yīng)用程序中需要多個(gè)UI 屏幕,也就需要多個(gè)Activity 類(lèi),并且在這些 Activity 之間進(jìn)行切換,這種切換就是通過(guò) Intent 機(jī)制來(lái)實(shí)現(xiàn)的。
在同一個(gè)應(yīng)用程序中切換 Activity時(shí),我們通常都知道要啟動(dòng)的 Activity 具體是哪一個(gè),因此常用顯式的 Intent 來(lái)實(shí)現(xiàn)。下面的例子用來(lái)實(shí)現(xiàn)一個(gè)非常簡(jiǎn)單的應(yīng)用程序 SimpleIntentTest ,它包括兩個(gè)UI 屏幕也就是兩個(gè) Activity——SimpleIntentTest類(lèi)和 TestActivity 類(lèi), SimpleIntentTest類(lèi)有一個(gè)按鈕用來(lái)啟動(dòng) TestActivity。
程序的代碼非常簡(jiǎn)單, SimpleIntentTest類(lèi)的源代碼如下:
Java代碼
package com.tope.samples.intent.simple;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
public class SimpleIntentTest extends Activity implements View.OnClickListener{
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super .onCreate(savedInstanceState);
setContentView(R.layout. main );
Button startBtn = (Button)findViewById(R.id. start_activity );
startBtn.setOnClickListener( this );
}
public void onClick(View v) {
switch (v.getId()) {
case R.id. start_activity :
Intent intent = new Intent( this , TestActivity. class );
startActivity(intent);
break ;
default :
break ;
}
}
}
上面的代碼中,主要是為“Start activity” 按鈕添加了 OnClickListener, 使得按鈕被點(diǎn)擊時(shí)執(zhí)行 onClick() 方法, onClick() 方法中則利用了 Intent 機(jī)制,來(lái)啟動(dòng) TestActivity,關(guān)鍵的代碼是下面這兩行:
Intent intent = new Intent( this , TestActivity. class );
startActivity(intent);
這里定義 Intent 對(duì)象時(shí)所用到的是 Intent 的構(gòu)造函數(shù)之一:
Intent ( Context packageContext, Class > cls)
兩個(gè)參數(shù)分別指定 Context 和 Class ,由于將Class 設(shè)置為 TestActivity.class,這樣便顯式的指定了TestActivity 類(lèi)作為該Intent 的接收者,通過(guò)后面的startActivity() 方法便可啟動(dòng) TestActivity 。
TestActivity 的代碼更為簡(jiǎn)單(定義 TestActivity類(lèi)需要新建 TestActivity.java 文件,如果你是一個(gè)初學(xué)者,你需要學(xué)會(huì)如何在 Eclipse 或其他開(kāi)發(fā)環(huán)境下添加一個(gè)新的類(lèi),這里不作詳述,請(qǐng)參考其他文檔),如下所示:
Java代碼
package com.tope.samples.intent.simple;
import android.app.Activity;
import android.os.Bundle;
public class TestActivity extends Activity {
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super .onCreate(savedInstanceState);
setContentView(R.layout. test_activity );
}
}
可見(jiàn) TestActivity僅僅是調(diào)用 setContentView 來(lái)顯示 test_activity.xml 中的內(nèi)容而已。對(duì)于 test_activity.xml及本例中所用到其他 xml 文件這里不作多余說(shuō)明。
如果我們僅僅是做上面的一些 工作,還不能達(dá)到利用 SimpleIntentTest 啟動(dòng) TestActivity的目的。事實(shí)上,這樣做會(huì)出現(xiàn)下面的 Exception ,導(dǎo)致程序退出。以下是利用 logcat 工具記錄的log 信息(省略了后半部分):
I/ActivityManager( 569): Displayed activity com.tope.samples/.SimpleIntentTest: 3018 ms
I/ActivityManager( 569): Starting activity: Intent { comp={com.tope.samples/com.tope.samples.TestActivity} }
D/AndroidRuntime( 932): Shutting down VM
W/dalvikvm( 932): threadid=3: thread exiting with uncaught exception (group=0x4000fe70)
E/AndroidRuntime( 932): Uncaught handler: thread main exiting due to uncaught exception
E/AndroidRuntime( 932): android.content.ActivityNotFoundException: Unable to find explicit activity class
{com.tope.samples/com.tope.samples.TestActivity}; have you declared this activity in your AndroidManifest.xml?
E/AndroidRuntime( 932): at android.app.Instrumentation.checkStartActivityResult(Instrumentation.java:1480)
E/AndroidRuntime( 932): at android.app.Instrumentation.execStartActivity(Instrumentation.java:1454)
E/AndroidRuntime( 932): at android.app.Activity.startActivityForResult(Activity.java:2656)
E/AndroidRuntime( 932): at android.app.Activity.startActivity(Activity.java:2700)
E/AndroidRuntime( 932): at com.tope.samples.SimpleIntentTest.onClick(SimpleIntentTest.java:24)
…
從這些log 中我們可以看到點(diǎn)擊按鈕后 startActivity 的調(diào)用過(guò)程,主要的原因是:“android.content.ActivityNotFoundException: Unable to find explicit activity class {com.tope.samples/com.tope.samples.TestActivity}; have you declared this activity in your AndroidManifest.xml?”
從這些log 我們可以看到原因是找不到 TestActivity這個(gè) Activity ,并且 log 中還給出了提示:你是否在AndroidManifest.xml 中聲明了這個(gè) Activity?解決問(wèn)題的方法也就是按照提示在 AndroidManifest.xml 中增加TestActivity 的聲明,如下所示:
XML/HTML代碼
xml version = "1.0" encoding = "utf-8" ?>
< manifest xmlns:android = "http://schemas.android.com/apk/res/android"
package = "com.tope.samples"
android:versionCode = "1"
android:versionName = "1.0" >
< application android:icon = "@drawable/icon" android:label ="@string/app_name" >
< activity android:name = ".SimpleIntentTest"
android:label = "@string/app_name" >
< intent-filter >
< action android:name = "android.intent.action.MAIN" />
< category android:name ="android.intent.category.LAUNCHER" />
intent-filter >
activity >
< activity android:name = ".TestActivity" />
application >
< uses-sdk android:minSdkVersion = "3" />
manifest >
完成這個(gè)修改后再重新運(yùn)行該程序,就一切都正常了。
從 AndroidManifest.xml修改的過(guò)程我們可以體會(huì)到, Intent 機(jī)制即使在程序內(nèi)部且顯式指定接收者,也還是需要在 AndroidManifest.xml 中聲明 TestActivity。這個(gè)過(guò)程并不像一個(gè)簡(jiǎn)單的函數(shù)調(diào)用,顯式的 Intent 也同樣經(jīng)過(guò)了Android 應(yīng)用程序框架所提供的支持,從滿足條件的 Activity 中進(jìn)行選擇,如果不在 AndroidManifest.xml中進(jìn)行聲明,則 Android 應(yīng)用程序框架找不到所需要的 Activity。
請(qǐng)讀者通過(guò)我們的示例來(lái)逐步理解 AndroidManifest.xml在這個(gè)過(guò)程中所扮演的角色,這樣有利于理解 Intent的作用 ,及后面的 Intent Filter。當(dāng)然,這個(gè)例子僅僅是開(kāi)始,且看下文分解 。
不同應(yīng)用程序之間的Activity切換
上面的例子我們所做的是在同 一應(yīng)用程序中進(jìn)行 Activity 的切換,那么在不同的應(yīng)用程序中,是否也能這么做呢,答案是肯定的,不過(guò)對(duì)應(yīng)的代碼要稍作修改。本例中我們需要兩個(gè)應(yīng)用程序,可利用上例中 的SimpleIntentTest作為其中之一,另外還需要寫(xiě)一個(gè)新的程序,來(lái)調(diào)用 SimpleIntentTest 應(yīng)用程序中的 TestActivity。
我們新建程序 CrossIntentTest(注意不是新建一個(gè)類(lèi),如果是 Eclipse 環(huán)境,選擇 File->New->Project新建工程),其中只有一個(gè) Activity ,其源代碼與 SimpleIntentTest.java 類(lèi)似 :
Java代碼
package com.tope.samples.intent.cross;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
public class CrossIntentTest extends Activity
implements View.OnClickListener{
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super .onCreate(savedInstanceState);
setContentView(R.layout. main );
Button startBtn = (Button)findViewById(R.id. start_activity );
startBtn.setOnClickListener( this );
}
public void onClick(View v) {
switch (v.getId()) {
case R.id. start_activity :
Intent intent = new Intent();
intent.setClassName( "com.tope.samples.intent.simple" ,
"com.tope.samples.intent.simple.TestActivity" );
startActivity(intent);
break ;
default :
break ;
}
}
}
注意比較它與 SimpleIntentTest的不同之處主要在于初始化 Intent 對(duì)象的過(guò)程:
Java代碼
Intent intent = new Intent();
intent.setClassName( "com.tope.samples.intent.simple" ,
"com.tope.samples.intent.simple.TestActivity" );
startActivity(intent);
這里采用了 Intent 最簡(jiǎn)單的不帶參數(shù)的構(gòu)造函數(shù) , 然后通過(guò) setClassName() 函數(shù)來(lái)指定要啟動(dòng)哪個(gè)包中的哪個(gè) Activity, 而不是像上例中的通過(guò) Intent ( Context packageContext, Class > cls) 這個(gè)構(gòu)造函數(shù)來(lái)初始化Intent 對(duì)象,這是因?yàn)椋獑?dòng)的 TestActivity 與 CrossIntentTest 不在同一個(gè)包中 , 要指定 Class 參數(shù)比較麻煩 , 所以通常啟動(dòng)不同程序的 Activity 時(shí)便采用上面的 setClassName() 的方式。除此之外,你也可以利用Android 提供的類(lèi)似的 setComponent() 方法,具體使用方法請(qǐng)參考 Android SDK的文檔。
另外我們還需要修改SimpleIntentTest 程序中的 AndroidManifest.xml 文件,為 TestActivity 的聲明添加Intent Filter ,即將原來(lái)的
XML/HTML代碼
修改為:
XML/HTML代碼
對(duì)于不同應(yīng)用之間的 Activity 的切換,這里需要在 Intent Filter中 設(shè)置至少一個(gè) Action,否則其他的應(yīng)用將沒(méi)有權(quán)限調(diào)用這個(gè) Activity 。這里我們開(kāi)始接觸 Intent Filter和 Action 這些概念了,讀者應(yīng)該可以感覺(jué)到,設(shè)置Intent Filter 和 Action 主要的目的,是為了讓其他需要調(diào)用這個(gè) Activity 的程序能夠順利的調(diào)用它。除了Action之外, Intent Filter 還可以設(shè)置 Category 、 Data等,用來(lái)更加精確的匹配 Intent 與 Activity。
隱式Intent(Implicit Intent)
如果 Intent 機(jī)制僅僅提供上面的顯式 Intent 用法的話,這種相對(duì)復(fù)雜的機(jī)制似乎意義并不是很大。確實(shí),Intent 機(jī)制更重要的作用在于下面這種隱式的 Intent ,即 Intent 的發(fā)送者不指定接收者,很可能不知道也不關(guān)心接收者是誰(shuí),而由 Android 框架去尋找最匹配的接收者。
最簡(jiǎn)單的隱式 Intent
我們先從最簡(jiǎn)單的例子開(kāi)始。 下面的 ImplicitIntentTest 程序用來(lái)啟動(dòng) Android 自帶的打電話功能的 Dialer 程序。
ImplicitIntentTest 程序只包含一個(gè)java 源文件 ImplicitIntentTest.java,代碼如下所示:
Java代碼
package com.tope.samples.intent.implicit;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
public class ImplicitIntentTest extends Activity
implements View.OnClickListener{
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super .onCreate(savedInstanceState);
setContentView(R.layout. main );
Button startBtn = (Button)findViewById(R.id. dial );
startBtn.setOnClickListener( this );
}
public void onClick(View v) {
switch (v.getId()) {
case R.id. dial :
Intent intent = new Intent(Intent. ACTION_DIAL );
startActivity(intent);
break ;
default :
break ;
}
}
}
該程序在Intent 的使用上,與上節(jié)中的使用方式有很大的不同,即根本不指定接收者,初始化 Intent 對(duì)象時(shí),只是傳入?yún)?shù),設(shè)定 Action為 Intent.ACTION_DIAL :
Intent intent = new Intent(Intent. ACTION_DIAL );
startActivity(intent);
這里使用的構(gòu)造函數(shù)的原型如下:
Intent ( String action);
這里讀者可暫時(shí)將action理解為描述這個(gè) Intent 的一種方式,這種使用方式看上去比較奇怪, Intent 的發(fā)送者只是指定了 Action為 Intent.ACTION_DIAL ,那么怎么找到接收者呢?來(lái)看下面的例子。
增加一個(gè)接收者
事實(shí)上接收者如果希望能夠接收某些 Intent ,需要像上節(jié)例子中一樣,通過(guò)在 AndroidManifest.xml中增加Activity 的聲明,并設(shè)置對(duì)應(yīng)的 Intent Filter 和 Action ,才能被 Android 的應(yīng)用程序框架所匹配。為了證明這一點(diǎn),我們修改上一 節(jié) SimpleIntentTest 程序中的 AndroidManifest.xml 文件,將 TestActivity 的聲明部分改為:
XML/HTML代碼
修改完之后注意要重新安裝 SimpleIntentTest 程序的apk 包,然后再?lài)L試運(yùn)行 ImplicitIntentTest 程序(不是SimpleIntentTest 程序)
這個(gè)截圖中的第二幅表示可以選擇 Dialer 或者 SimpleIntentTest 程序來(lái)完成 Intent.ACTION_DIAL ,也就是說(shuō),針對(duì) Intent.ACTION_DIAL, Android 框架找到了兩個(gè)符合條件的 Activity,因此它將這兩個(gè) Activity 分別列出,供用戶選擇。
回過(guò)頭來(lái)看我們是怎么做到這一點(diǎn)的。我們僅僅在 SimpleIntentTest 程序的 AndroidManifest.xml 文件中增加了下面的兩行:
XML/HTML代碼
category android:name = "android.intent.category.DEFAULT"/>
這兩行修改了原來(lái)的 Intent Filter,這樣這個(gè) Activity 才能夠接收到我們發(fā)送的 Intent 。我們通過(guò)這個(gè)改動(dòng)及其作用,可以進(jìn)一步理解隱式 Intent, Intent Filter 及 Action, Category 等概念—— Intent 發(fā)送者設(shè)定 Action 來(lái)說(shuō)明將要進(jìn)行的動(dòng)作,而 Intent 的接收者在 AndroidManifest.xml 文件中通過(guò)設(shè)定 Intent Filter來(lái)聲明自己能接收哪些Intent 。
到此,關(guān)于“Android中的Intent機(jī)制有什么用”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)?lái)更多實(shí)用的文章!