編輯:關於Android編程
【正文】
Intent組件雖然不是四大組件,但卻是連接四大組件的橋梁,學習好這個知識,也非常的重要。
一、什麼是Intent
1、Intent的概念:
activity、service和broadcast receiver之間是通過Intent進行通信的,而另外一個組件Content Provider本身就是一種通信機制,不需要通過Intent。我們來看下面這個圖就知道了:
如果Activity1需要和Activity2進行聯系,二者不需要直接聯系,而是通過Intent作為橋梁。通俗來講,Intnet類似於中介、媒婆的角色。
2、對於向這三種組件發送intent有不同的機制:
使用Context.startActivity() 或 Activity.startActivityForResult(),傳入一個intent來啟動一個activity。使用 Activity.setResult(),傳入一個intent來從activity中返回結果。二、Intent的相關屬性:
Intent類型分為顯式Intent(直接類型)、隱式Intent(間接類型)。官方建議使用隱式Intent。上述屬性中,component屬性為直接類型,其他均為間接類型。
相比與顯式Intent,隱式Intnet則含蓄了許多,它並不明確指出我們想要啟動哪一個活動,而是指定一系列更為抽象的action和category等信息,然後交由系統去分析這個Intent,並幫我們找出合適的活動去啟動。
Activity 中 Intent Filter 的匹配過程 :
1、component(組件):目的組件
Component屬性明確指定Intent的目標組件的類名稱。(屬於直接Intent)
如果 component這個屬性有指定的話,將直接使用它指定的組件。指定了這個屬性以後,Intent的其它所有屬性都是可選的。
例如,啟動第二個Activity時,我們可以這樣來寫:
1 button1.setOnClickListener(new OnClickListener() { 2 @Override 3 public void onClick(View v) { 4 //創建一個意圖對象 5 Intent intent = new Intent(); 6 //創建組件,通過組件來響應 7 ComponentName component = new ComponentName(MainActivity.this, SecondActivity.class); 8 intent.setComponent(component); 9 startActivity(intent); 10 } 11 });
如果寫的簡單一點,監聽事件onClick()方法裡可以這樣寫:
1 Intent intent = new Intent(); 2 //setClass函數的第一個參數是一個Context對象 3 //Context是一個類,Activity是Context類的子類,也就是說,所有的Activity對象,都可以向上轉型為Context對象 4 //setClass函數的第二個參數是一個Class對象,在當前場景下,應該傳入需要被啟動的Activity類的class對象 5 intent.setClass(MainActivity.this, SecondActivity.class); 6 startActivity(intent);
再簡單一點,可以這樣寫:(當然,也是最常見的寫法)
1 Intent intent = new Intent(MainActivity.this,SecondActivity.class); 2 startActivity(intent);
2、Action(動作):用來表現意圖的行動
當日常生活中,描述一個意願或願望的時候,總是有一個動詞在其中。比如:我想“做”三個俯臥撐;我要“寫” 一封情書,等等。在Intent中,Action就是描述做、寫等動作的,當你指明了一個Action,執行者就會依照這個動作的指示,接受相關輸入,表現對應行為,產生符合的輸出。在Intent類中,定義了一批量的動作,比如ACTION_VIEW,ACTION_PICK等, 基本涵蓋了常用動作。加的動作越多,越精確。
Action 是一個用戶定義的字符串,用於描述一個 Android 應用程序組件,一個 Intent Filter 可以包含多個 Action。在 AndroidManifest.xml 的Activity 定義時,可以在其
3、category(類別):用來表現動作的類別
Category屬性也是作為
Action 和category通常是放在一起用的,所以這裡一起介紹一下。我們來先來舉一個例子:
新建一個工程文件smyh006_Intent01,在默認文件的基礎之上,新建文件SecondActicity.java和activity_second.xml。
緊接著,我們要到清單文件中進行注冊,打開AndroidManifest.xml,添加SecondActivity的action和category的過濾器:
1 34 5 76
上方代碼,表示SecondActicity可以匹配第4行的MY_ACTION這個動作,此時,如果在其他的Acticity通過這個action的條件來查找,那SecondActicity就具備了這個條件。類似於相親時,我要求對方有哪些條件,然後對方這個SecondActicity恰巧滿足了這個條件(夠通俗了吧)。
注:如果沒有指定的category,則必須使用默認的DEFAULT(即上方第5行代碼)。
也就是說:只有和
現在來修改MainActivity.java中按鈕的點擊事件,代碼如下:
1 button1.setOnClickListener(new OnClickListener() { 2 @Override 3 public void onClick(View v) { 4 //啟動另一個Activity,(通過action屬性進行查找) 5 Intent intent = new Intent(); 6 //設置動作(實際action屬性就是一個字符串標記而已) 7 intent.setAction("com.example.smyh006intent01.MY_ACTION"); //方法:Intent android.content.Intent.setAction(String action) 8 startActivity(intent); 9 } 10 });
上方代碼中,也可以換成下面這種簡潔的方式:
1 button1.setOnClickListener(new OnClickListener() { 2 @Override 3 public void onClick(View v) { 4 //啟動另一個Activity,(通過action屬性進行查找) 5 Intent intent = new Intent("com.example.smyh006intent01.MY_ACTION");//方法: android.content.Intent.Intent(String action) 6 startActivity(intent); 7 } 8 });
上方第5行代碼:在這個Intent中,我並沒有指定具體哪一個Activity,我只是指定了一個action的常量。所以說,隱式Intent的作用就表現的淋漓盡致了。此時,點擊MainActicity中的按鈕,就會跳到SecondActicity中去。
上述情況只有SecondActicity匹配成功。如果有多個組件匹配成功,就會以對話框列表的方式讓用戶進行選擇。我們來詳細介紹一下:
我們新建文件ThirdActicity.java和activity_third.xml,然後在清單文件AndroidManifest.xml中添加ThirdActivity的action和category的過濾器:
1 34 5 76
此時,運行程序,當點擊MainActivity中的按鈕時,彈出如下界面:
相信大家看到了這個界面,應該就一目了然了。於是我們可以做出如下總結:
在自定義動作時,使用activity組件時,必須添加一個默認的類別
具體的實現為:
如果有多個組件被匹配成功,就會以對話框列表的方式讓用戶進行選擇。
每個Intent中只能指定一個action,但卻能指定多個category;類別越多,動作越具體,意圖越明確(類似於相親時,給對方提了很多要求)。
目前我們的Intent中只有一個默認的category,現在可以通過intent.addCategory()方法來實現。修改MainActivity中按鈕的點擊事件,代碼如下:
1 button1.setOnClickListener(new OnClickListener() { 2 @Override 3 public void onClick(View v) { 4 //啟動另一個Activity,(通過action屬性進行查找) 5 Intent intent = new Intent(); 6 //設置動作(實際action屬性就是一個字符串標記而已) 7 intent.setAction("com.example.smyh006intent01.MY_ACTION"); //方法:Intent android.content.Intent.setAction(String action) 8 intent.addCategory("com.example.smyh006intent01.MY_CATEGORY"); 9 startActivity(intent); 10 } 11 });
既然在Intent中增加了一個category,那麼我們要在清單文件中去聲明這個category,不然程序將無法運行。代碼如下:
1 android:name=".SecondActivity"> 23 4 75 6
此時,點擊MainActicity中的按鈕,就會跳到SecondActicity中去。
總結如下:
自定義類別:在Intent添加類別可以添加多個類別,那就要求被匹配的組件必須同時滿足這多個類別,才能匹配成功。操作Activity的時候,如果沒有類別,須加上默認類別
4、data(數據):表示與動作要操縱的數據
Data是用一個uri對象來表示的,uri代表數據的地址,屬於一種標識符。通常情況下,我們使用action+data屬性的組合來描述一個意圖:做什麼
使用隱式Intent,我們不僅可以啟動自己程序內的活動,還可以啟動其他程序的活動,這使得Android多個應用程序之間的功能共享成為了可能。比如應用程序中需要展示一個網頁,沒有必要自己去實現一個浏覽器(事實上也不太可能),而是只需要條用系統的浏覽器來打開這個網頁就行了。
【實例】打開指定網頁:
MainActivity.java中,監聽器部分的核心代碼如下:
1 button1.setOnClickListener(new OnClickListener() { 2 @Override 3 public void onClick(View v) { 4 Intent intent = new Intent(); 5 intent.setAction(Intent.ACTION_VIEW); 6 Uri data = Uri.parse("http://www.baidu.com"); 7 intent.setData(data); 8 startActivity(intent); 9 } 10 });
當然,上方代碼也可以簡寫成:
1 button1.setOnClickListener(new OnClickListener() { 2 @Override 3 public void onClick(View v) { 4 Intent intent = new Intent(Intent.ACTION_VIEW); 5 intent.setData(Uri.parse("http://www.baidu.com")); 6 startActivity(intent); 7 } 8 });
第4行代碼:指定了Intent的action是Intent.ACTION_VIEW,表示查看的意思,這是一個Android系統內置的動作;
第5行代碼:通過Uri.parse()方法,將一個網址字符串解析成一個Uri對象,再調用intent的setData()方法將這個Uri對象傳遞進去。
當點擊按鈕時,將跳到如下界面:
此時, 調用的是系統默認的浏覽器,也就是說,只調用了這一個組件。現在如果有多個組件得到了匹配,應該是什麼情況呢?
我們修改修改清單文件中對SecondAcivity的聲明:
1 34 5 86 7
現在,SecondActivity也匹配成功了,我們運行程序,點擊MainActicity的按鈕時,彈出如下界面供我們選擇:
我們可以總結如下:
注:系統默認的浏覽器並沒有做出優先級聲明,其優先級默認為正數。
優先級的配置如下:
在清單文件中修改對SecondAcivity的聲明,即增加一行代碼,通過來android:priority設置優先級,如下:
1 34 5 86 7
注:
Data屬性的聲明中要指定訪問數據的Uri和MIME類型。可以在元素中通過一些屬性來設置:
android:scheme、android:path、android:port、android:mimeType、android:host等,通過這些屬性來對應一個典型的Uri格式scheme://host:port/path。例如:http://www.google.com。
5、type(數據類型):對於data范例的描寫
如果Intent對象中既包含Uri又包含Type,那麼,在
Type屬性用於明確指定Data屬性的數據類型或MIME類型,但是通常來說,當Intent不指定Data屬性時,Type屬性才會起作用,否則Android系統將會根據Data屬性值來分析數據的類型,所以無需指定Type屬性。
data和type屬性一般只需要一個,通過setData方法會把type屬性設置為null,相反設置setType方法會把data設置為null,如果想要兩個屬性同時設置,要使用Intent.setDataAndType()方法。
【任務】:data+type屬性的使用【實例】:播放指定路徑的mp3文件。
具體如下:
新建工程文件smyh006_Intent02,MainActivity.java中按鈕監聽事件部分的代碼如下:
1 button.setOnClickListener(new OnClickListener(){ 2 @Override 3 public void onClick(View v) { 4 Intent intent = new Intent(); 5 intent.setAction(Intent.ACTION_VIEW); 6 Uri data = Uri.parse("file:///storage/sdcard0/平凡之路.mp3"); 7 //設置data+type屬性 8 intent.setDataAndType(data, "audio/mp3"); //方法:Intent android.content.Intent.setDataAndType(Uri data, String type) 9 startActivity(intent); 10 } 11 });
代碼解釋:
第6行:"file://"表示查找文件,後面再加上我的小米手機存儲卡的路徑:/storage/sdcard0,再加上具體歌曲的路徑。
第8行:設置data+type屬性
運行後,當點擊按鈕時,效果如下:
上方界面中,使用的是小米系統默認的音樂播放器。
6、extras(擴展信息):擴展信息
是其它所有附加信息的集合。使用extras可以為組件提供擴展信息,比如,如果要執行“發送電子郵件”這個
動作,可以將電子郵件的標題、正文等保存在extras裡,傳給電子郵件發送組件。
7、Flags(標志位):期望這個意圖的運行模式
一個程序啟動後系統會為這個程序分配一個task供其使用,另外同一個task裡面可以擁有不同應用程序的activity。那麼,同一個程序能不能擁有多個task?這就涉及到加載activity的啟動模式,這個需要單獨講一下。
注:android中一組邏輯上在一起的activity被叫做task,自己認為可以理解成一個activity堆棧。
三、Activity的啟動模式:(面試注意)
Activity有四種啟動模式:standard、singleTop、singleTask、singleInstance。可以在AndroidManifest.xml中activity標簽的屬性android:launchMode中設置該activity的加載模式。
注:如果以singleTop模式啟動的activity的一個實例已經存在於返回桟中,但是不在桟頂,那麼它的行為和standard模式相同,也會創建多個實例;
注:前面三種模式中,每個應用程序都有自己的返回棧,同一個活動在不同的返回棧中入棧時,必然是創建了新的實例。而使用singleInstance模式可以解決這個問題,在這種模式下會有一個單獨的返回棧來管理這個活動,不管是哪一個應用程序來訪問這個活動,都公用同一個返回棧,也就解決了共享活動實例的問題。(此時可以實現任務之間的切換,而不是單獨某個棧中的實例切換)
其實我們不在清單文件中設置,只在代碼中通過flag來設置也是可以的,如下:
1 Intent intent = new Intent(MainActivity.this,SecondActivity.class); 2 //相當於singleTask 3 intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); 4 startActivity(intent);
1 Intent intent = new Intent(MainActivity.this,SecondActivity.class); 2 //相當於singleTop 3 intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); 4 startActivity(intent);
三、Intent的常見應用:
1、打開指定網頁:(直接復制的上面的代碼)
MainActivity.java中,監聽器部分的核心代碼如下:
1 button1.setOnClickListener(new OnClickListener() { 2 @Override 3 public void onClick(View v) { 4 Intent intent = new Intent(); 5 intent.setAction(Intent.ACTION_VIEW);//方法:android.content.Intent.Intent(String action) 6 Uri data = Uri.parse("http://www.baidu.com"); 7 intent.setData(data); 8 startActivity(intent); 9 } 10 });
當然,上方代碼也可以簡寫成:
1 button1.setOnClickListener(new OnClickListener() { 2 @Override 3 public void onClick(View v) { 4 Intent intent = new Intent(Intent.ACTION_VIEW); 5 intent.setData(Uri.parse("http://www.baidu.com")); 6 startActivity(intent); 7 } 8 });
第4行代碼:指定了Intent的action是Intent.ACTION_VIEW,表示查看的意思,這是一個Android系統內置的動作;
第5行代碼:通過Uri.parse()方法,將一個網址字符串解析成一個Uri對象,再調用intent的setData()方法將這個Uri對象傳遞進去。
或者可以寫成:
1 button1.setOnClickListener(new OnClickListener() { 2 @Override 3 public void onClick(View v) { 4 Uri uri = Uri.parse("http://www.baidu.com"); 5 Intent intent = new Intent(Intent.ACTION_VIEW,uri);//方法: android.content.Intent.Intent(String action, Uri uri) 6 startActivity(intent); 7 } 8 });
2、打電話:
【方式一】打開撥打電話的界面:
1 Intent intent = new Intent(Intent.ACTION_DIAL); 2 intent.setData(Uri.parse("tel:10086")); 3 startActivity(intent);
運行程序後,點擊按鈕,顯示如下界面:
【方式二】直接撥打電話:
1 Intent intent = new Intent(Intent.ACTION_CALL); 2 intent.setData(Uri.parse("tel:10086")); 3 startActivity(intent);
要使用這個功能必須在配置文件中加入權限:(加一行代碼)
14
3、發送短信:
【方式一】打開發送短信的界面:action+type
1 Intent intent = new Intent(Intent.ACTION_VIEW); 2 intent.setType("vnd.android-dir/mms-sms"); 3 intent.putExtra("sms_body", "具體短信內容"); //"sms_body"為固定內容 4 startActivity(intent);
【方式二】打開發短信的界面(同時指定電話號碼):action+data
1 Intent intent = new Intent(Intent.ACTION_SENDTO); 2 intent.setData(Uri.parse("smsto:18780260012")); 3 intent.putExtra("sms_body", "具體短信內容"); //"sms_body"為固定內容 4 startActivity(intent);
4、播放指定路徑音樂:action+data+type
1 Intent intent = new Intent(Intent.ACTION_VIEW); 2 Uri uri = Uri.parse("file:///storage/sdcard0/平凡之路.mp3"); ////路徑也可以寫成:"/storage/sdcard0/平凡之路.mp3" 3 intent.setDataAndType(uri, "audio/mp3"); //方法:Intent android.content.Intent.setDataAndType(Uri data, String type) 4 startActivity(intent);
5、卸載程序:action+data(例如點擊按鈕,卸載某個應用程序,根據包名來識別)
注:無論是安裝還是卸載,應用程序是根據包名package來識別的。
1 Intent intent = new Intent(Intent.ACTION_DELETE); 2 Uri data = Uri.parse("package:com.example.smyh006intent01"); 3 intent.setData(data); 4 startActivity(intent);
6、安裝程序:action+data+type
1 Intent intent = new Intent(Intent.ACTION_VIEW); 2 Uri data = Uri.fromFile(new File("/storage/sdcard0/AndroidTest/smyh006_Intent01.apk")); //路徑不能寫成:"file:///storage/sdcard0/···" 3 intent.setDataAndType(data, "application/vnd.android.package-archive"); //Type的字符串為固定內容 4 startActivity(intent);
注:第2行的路徑不能寫成:"file:///storage/sdcard0/···",不然報錯如下:
疑問:通過下面的這種方式安裝程序,運行時為什麼會出錯呢?
1 //通過指定的action來安裝程序 2 public void installClickTwo(View view){ 3 Intent intent = new Intent(Intent.ACTION_PACKAGE_ADDED); 4 Uri data = Uri.fromFile(new File("/storage/sdcard0/AndroidTest/smyh006_Intent01.apk")); //路徑不能寫成:"file:///storage/sdcard0/···" 5 intent.setData(data); 6 startActivity(intent); 7 }
綜上所述,完整版代碼如下:
111
MainActivity.java代碼如下:
1 package com.example.m06intent01; 2 import java.io.File; 3 import android.app.Activity; 4 import android.content.Intent; 5 import android.net.Uri; 6 import android.os.Bundle; 7 import android.view.Menu; 8 import android.view.View; 9 public class MainActivity extends Activity { 10 @Override 11 protected void onCreate(Bundle savedInstanceState) { 12 super.onCreate(savedInstanceState); 13 setContentView(R.layout.activity_main); 14 } 15 //打開指定網頁 16 public void browsePageClick(View view){ 17 Intent intent = new Intent(Intent.ACTION_VIEW); 18 intent.setData(Uri.parse("http://www.baidu.com/")); 19 startActivity(intent); 20 21 } 22 23 //打開撥號面板 24 public void openDialPageClick(View view){ 25 Intent intent = new Intent(Intent.ACTION_DIAL); 26 intent.setData(Uri.parse("tel:10086")); 27 startActivity(intent); 28 } 29 30 //直接撥打指定號碼 31 public void dialPhoneClick(View view){ 32 Intent intent = new Intent(Intent.ACTION_CALL); 33 intent.setData(Uri.parse("tel:10086")); 34 startActivity(intent); 35 } 36 37 //打開發短信的界面:action+type 38 public void openMsgPageClick(View view){ 39 Intent intent = new Intent(Intent.ACTION_VIEW); 40 intent.setType("vnd.android-dir/mms-sms"); 41 intent.putExtra("sms_body", "具體短信內容"); //"sms_body"為固定內容 42 startActivity(intent); 43 } 44 45 //打開發短信的界面(指定電話號碼):action+data 46 public void sendMsgClick(View view){ 47 Intent intent = new Intent(Intent.ACTION_SENDTO); 48 intent.setData(Uri.parse("smsto:18780260012")); 49 intent.putExtra("sms_body", "具體短信內容"); //"sms_body"為固定內容 50 startActivity(intent); 51 } 52 53 //播放指定路徑音樂 54 public void playMusicClick(View view){ 55 Intent intent = new Intent(Intent.ACTION_VIEW); 56 Uri uri = Uri.parse("file:///storage/sdcard0/平凡之路.mp3"); //路徑也可以寫成:"/storage/sdcard0/平凡之路.mp3" 57 intent.setDataAndType(uri, "audio/mp3"); //方法:Intent android.content.Intent.setDataAndType(Uri data, String type) 58 startActivity(intent); 59 } 60 61 //卸載某個應用程序,根據包名來識別 62 public void uninstallClick(View view){ 63 Intent intent = new Intent(Intent.ACTION_DELETE); 64 Uri data = Uri.parse("package:com.example.smyh006intent01"); 65 intent.setData(data); 66 startActivity(intent); 67 } 68 69 //安裝某個應用程序,根據apk的文件名來識別 70 public void installClick(View view){ 71 Intent intent = new Intent(Intent.ACTION_VIEW); 72 Uri data = Uri.fromFile(new File("/storage/sdcard0/AndroidTest/smyh006_Intent01.apk")); //路徑不能寫成:"file:///storage/sdcard0/···" 73 intent.setDataAndType(data, "application/vnd.android.package-archive"); //Type的字符串為固定內容 74 startActivity(intent); 75 } 76 77 78 @Override 79 public boolean onCreateOptionsMenu(Menu menu) { 80 // Inflate the menu; this adds items to the action bar if it is present. 81 getMenuInflater().inflate(R.menu.main, menu); 82 return true; 83 } 84 85 }
運行後,主界面如下:
WeTest導讀說起Android的自動化測試,相信有很多小伙伴都接觸過或者有所耳聞,本文從框架最基本的功能介紹及API的使用入手,結合簡單的項目實戰來幫忙大家對該框架進
首先理解Android 的 egl 的函數調用,先要得到display (對應顯示器),而後選擇一個支持的config(這是顯卡支持的),而後創建WindowSurfac
本文實例講述了Android ScrollView只能添加一個子控件問題解決方法。分享給大家供大家參考,具體如下:有下面一段代碼<?xml version=
讓我們一起學習一下模擬器的使用。本文內容如下: 模擬器和真機的比較 創建Android模擬器(emulator) 運行Android模擬器 設置簡體中文語言界面