Android Activity和Intent機(jī)制學(xué)習(xí)筆記
Activity
Android中,Activity是所有程序的根本,所有程序的流程都運(yùn)行在Activity之中,Activity具有自己的生命周期(見http://www.cnblogs.com/feisky/archive/2010/01/01/1637427.html,由系統(tǒng)控制生命周期,程序無法改變,但可以用onSaveInstanceState保存其狀態(tài))。
對(duì)于Activity,關(guān)鍵是其生命周期的把握(如下圖),其次就是狀態(tài)的保存和恢復(fù)(onSaveInstanceState onRestoreInstanceState),以及Activity之間的跳轉(zhuǎn)和數(shù)據(jù)傳輸(intent)。
Activity中常用的函數(shù)有SetContentView() findViewById() finish() startActivity(),其生命周期涉及的函數(shù)有:
- void onCreate(Bundle savedInstanceState)
- void onStart()
- void onRestart()
- void onResume()
- void onPause()
- void onStop()
- void onDestroy()
注意的是,Activity的使用需要在Manifest文件中添加相應(yīng)的<Activity>,并設(shè)置其屬性和intent-filter。
Intent
Android中提供了Intent機(jī)制來協(xié)助應(yīng)用間的交互與通訊,Intent負(fù)責(zé)對(duì)應(yīng)用中一次操作的動(dòng)作、動(dòng)作涉及數(shù)據(jù)、附加數(shù)據(jù)進(jìn)行描述,Android則根據(jù)此Intent的描述,負(fù)責(zé)找到對(duì)應(yīng)的組件,將 Intent傳遞給調(diào)用的組件,并完成組件的調(diào)用。Intent不僅可用于應(yīng)用程序之間,也可用于應(yīng)用程序內(nèi)部的Activity/Service之間的交互。因此,Intent在這里起著一個(gè)媒體中介的作用,專門提供組件互相調(diào)用的相關(guān)信息,實(shí)現(xiàn)調(diào)用者與被調(diào)用者之間的解耦。在SDK中給出了Intent作用的表現(xiàn)形式為:
通過Context.startActivity() orActivity.startActivityForResult() 啟動(dòng)一個(gè)Activity;
通過 Context.startService() 啟動(dòng)一個(gè)服務(wù),或者通過Context.bindService() 和后臺(tái)服務(wù)交互;
通過廣播方法(比如 Context.sendBroadcast(),Context.sendOrderedBroadcast(), Context.sendStickyBroadcast()) 發(fā)給broadcast receivers。
Intent屬性的設(shè)置,包括以下幾點(diǎn):(以下為XML中定義,當(dāng)然也可以通過Intent類的方法來獲取和設(shè)置)
(1)Action,也就是要執(zhí)行的動(dòng)作
SDk中定義了一些標(biāo)準(zhǔn)的動(dòng)作,包括
當(dāng)然,也可以自定義動(dòng)作(自定義的動(dòng)作在使用時(shí),需要加上包名作為前綴,如"com.example.project.SHOW_COLOR”),并可定義相應(yīng)的Activity來處理我們的自定義動(dòng)作。
(2)Data,也就是執(zhí)行動(dòng)作要操作的數(shù)據(jù)
Android中采用指向數(shù)據(jù)的一個(gè)URI來表示,如在聯(lián)系人應(yīng)用中,一個(gè)指向某聯(lián)系人的URI可能為:content://contacts/1。對(duì)于不同的動(dòng)作,其URI數(shù)據(jù)的類型是不同的(可以設(shè)置type屬性指定特定類型數(shù)據(jù)),如ACTION_EDIT指定Data為文件URI,打電話為tel:URI,訪問網(wǎng)絡(luò)為http:URI,而由content provider提供的數(shù)據(jù)則為content: URIs。
(3)type(數(shù)據(jù)類型),顯式指定Intent的數(shù)據(jù)類型(MIME)。一般Intent的數(shù)據(jù)類型能夠根據(jù)數(shù)據(jù)本身進(jìn)行判定,但是通過設(shè)置這個(gè)屬性,可以強(qiáng)制采用顯式指定的類型而不再進(jìn)行推導(dǎo)。
(4)category(類別),被執(zhí)行動(dòng)作的附加信息。例如 LAUNCHER_CATEGORY 表示Intent 的接受者應(yīng)該在Launcher中作為頂級(jí)應(yīng)用出現(xiàn);而ALTERNATIVE_CATEGORY表示當(dāng)前的Intent是一系列的可選動(dòng)作中的一個(gè),這些動(dòng)作可以在同一塊數(shù)據(jù)上執(zhí)行。還有其他的為
Constant Meaning
(5)component(組件),指定Intent的的目標(biāo)組件的類名稱。通常 Android會(huì)根據(jù)Intent 中包含的其它屬性的信息,比如action、data/type、category進(jìn)行查找,最終找到一個(gè)與之匹配的目標(biāo)組件。但是,如果 component這個(gè)屬性有指定的話,將直接使用它指定的組件,而不再執(zhí)行上述查找過程。指定了這個(gè)屬性以后,Intent的其它所有屬性都是可選的。
(6)extras(附加信息),是其它所有附加信息的集合。使用extras可以為組件提供擴(kuò)展信息,比如,如果要執(zhí)行“發(fā)送電子郵件”這個(gè)動(dòng)作,可以將電子郵件的標(biāo)題、正文等保存在extras里,傳給電子郵件發(fā)送組件。
理解Intent的關(guān)鍵之一是理解清楚Intent的兩種基本用法:一種是顯式的Intent,即在構(gòu)造Intent對(duì)象時(shí)就指定接收者;另一種是隱式的Intent,即Intent的發(fā)送者在構(gòu)造Intent對(duì)象時(shí),并不知道也不關(guān)心接收者是誰,有利于降低發(fā)送者和接收者之間的耦合。
對(duì)于顯式Intent,Android不需要去做解析,因?yàn)槟繕?biāo)組件已經(jīng)很明確,Android需要解析的是那些隱式Intent,通過解析,將 Intent映射給可以處理此Intent的Activity、IntentReceiver或Service。
Intent解析機(jī)制主要是通過查找已注冊(cè)在AndroidManifest.xml中的所有IntentFilter及其中定義的Intent,最終找到匹配的Intent。在這個(gè)解析過程中,Android是通過Intent的action、type、category這三個(gè)屬性來進(jìn)行判斷的,判斷方法如下:
如果Intent指明定了action,則目標(biāo)組件的IntentFilter的action列表中就必須包含有這個(gè)action,否則不能匹配;
如果Intent沒有提供type,系統(tǒng)將從data中得到數(shù)據(jù)類型。和action一樣,目標(biāo)組件的數(shù)據(jù)類型列表中必須包含Intent的數(shù)據(jù)類型,否則不能匹配。
如果Intent中的數(shù)據(jù)不是content: 類型的URI,而且Intent也沒有明確指定它的type,將根據(jù)Intent中數(shù)據(jù)的scheme (比如 http: 或者mailto:) 進(jìn)行匹配。同上,Intent 的scheme必須出現(xiàn)在目標(biāo)組件的scheme列表中。
如果Intent指定了一個(gè)或多個(gè)category,這些類別必須全部出現(xiàn)在組建的類別列表中。比如Intent中包含了兩個(gè)類別:LAUNCHER_CATEGORY 和 ALTERNATIVE_CATEGORY,解析得到的目標(biāo)組件必須至少包含這兩個(gè)類別。
Intent-Filter的定義
一些屬性設(shè)置的例子:
- <action android:name="com.example.project.SHOW_CURRENT" />
- <category android:name="android.intent.category.DEFAULT" />
- <data android:mimeType="video/mpeg" android:scheme="http" . . . />
- <data android:mimeType="image/*" />
- <data android:scheme="http" android:type="video/*" />
完整的實(shí)例
- <activity android:name="NotesList" android:label="@string/title_notes_list">
- <intent-filter>
- <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- <intent-filter>
- <action android:name="android.intent.action.VIEW" />
- <action android:name="android.intent.action.EDIT" /> <action android:name="android.intent.action.PICK" /> <category
- android:name="android.intent.category.DEFAULT" />
- <data android:mimeType="vnd.android.cursor.dir/vnd.google.note" />
- </intent-filter>
- <intent-filter>
- <action android:name="android.intent.action.GET_CONTENT" /> <category
- android:name="android.intent.category.DEFAULT" /> <data android:mimeType="vnd.android.cursor.item/vnd.google.note" />
- </intent-filter>
- </activity>
#p#
Intent用法實(shí)例
1.無參數(shù)Activity跳轉(zhuǎn)
- Intent it = new Intent(Activity.Main.this, Activity2.class);
- startActivity(it);
2.向下一個(gè)Activity傳遞數(shù)據(jù)(使用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);
對(duì)于數(shù)據(jù)的獲取可以采用:
- Bundle bundle=getIntent().getExtras();
- String name=bundle.getString("name");
3.向上一個(gè)Activity返回結(jié)果(使用setResult,針對(duì)startActivityForResult(it,REQUEST_CODE)啟動(dòng)的Activity)
- Bundle bundle=getIntent().getExtras();
- String name=bundle.getString("name");
4.回調(diào)上一個(gè)Activity的結(jié)果處理函數(shù)(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);
- }
- }
- }
下面是轉(zhuǎn)載來的其他的一些Intent用法實(shí)例(轉(zhuǎn)自javaeye)
顯示網(wǎng)頁
- 1. Uri uri = Uri.parse("http://google.com");
- 2. Intent it = new Intent(Intent.ACTION_VIEW, uri);
- 3. startActivity(it);
顯示地圖
- Uri uri = Uri.parse("http://google.com");
- Intent it = new Intent(Intent.ACTION_VIEW, uri);
- startActivity(it);
路徑規(guī)劃
- Uri uri = Uri.parse("http://maps.google.com/maps?f=d&saddr=startLat%20startLng&daddr=endLat%20endLng&hl=en");
- Intent it = new Intent(Intent.ACTION_VIEW, uri);
- startActivity(it);
- //where startLat, startLng, endLat, endLng are a long with 6 decimals like: 50.123456
打電話
- //叫出撥號(hào)程序
- Uri uri = Uri.parse("tel:0800000123");
- Intent it = new Intent(Intent.ACTION_DIAL, uri);
- startActivity(it);
//直接打電話出去
- Uri uri = Uri.parse("tel:0800000123");
- Intent it = new Intent(Intent.ACTION_CALL, uri);
- startActivity(it);
- //用這個(gè),要在 AndroidManifest.xml 中,加上
- //<uses-permission id="android.permission.CALL_PHONE" />
傳送SMS/MMS
- //調(diào)用短信程序
- Intent it = new Intent(Intent.ACTION_VIEW, uri);
- it.putExtra("sms_body", "The SMS text");
- it.setType("vnd.android-dir/mms-sms");
- startActivity(it);
- //傳送消息
- Uri uri = Uri.parse("smsto://0800000123");
- Intent it = new Intent(Intent.ACTION_SENDTO, uri);
- it.putExtra("sms_body", "The SMS text");
- startActivity(it);
- //傳送 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);
傳送 Email
- Uri uri = Uri.parse("mailto:xxx@abc.com");
- Intent it = new Intent(Intent.ACTION_SENDTO, uri);
- startActivity(it);
- Intent it = new Intent(Intent.ACTION_SEND);
- it.putExtra(Intent.EXTRA_EMAIL, "me@abc.com");
- it.putExtra(Intent.EXTRA_TEXT, "The email body text");
- it.setType("text/plain");
- startActivity(Intent.createChooser(it, "Choose Email Client"));
- Intent it=new Intent(Intent.ACTION_SEND);
- String[] tos={"me@abc.com"};
- String[] ccs={"you@abc.com"};
- it.putExtra(Intent.EXTRA_EMAIL, tos);
- it.putExtra(Intent.EXTRA_CC, ccs);
- it.putExtra(Intent.EXTRA_TEXT, "The email body text");
- it.putExtra(Intent.EXTRA_SUBJECT, "The email subject text");
- it.setType("message/rfc822");
- startActivity(Intent.createChooser(it, "Choose Email Client"));
- //傳送附件
- Intent it = new Intent(Intent.ACTION_SEND);
- it.putExtra(Intent.EXTRA_SUBJECT, "The email subject text");
- it.putExtra(Intent.EXTRA_STREAM, "file:///sdcard/mysong.mp3");
- sendIntent.setType("audio/mp3");
- startActivity(Intent.createChooser(it, "Choose Email Client"));
播放多媒體
- Uri uri = Uri.parse("file:///sdcard/song.mp3");
- Intent it = new Intent(Intent.ACTION_VIEW, uri);
- it.setType("audio/mp3");
- startActivity(it);
- Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, "1");
- Intent it = new Intent(Intent.ACTION_VIEW, uri);
- startActivity(it);
- Market 相關(guān)
- //尋找某個(gè)應(yīng)用
- Uri uri = Uri.parse("market://search?q=pname:pkg_name");
- Intent it = new Intent(Intent.ACTION_VIEW, uri); startActivity(it);
- //where pkg_name is the full package path for an application
- /顯示某個(gè)應(yīng)用的相關(guān)信息
- Uri uri = Uri.parse("market://details?id=app_id");
- Intent it = new Intent(Intent.ACTION_VIEW, uri); startActivity(it);
- /where app_id is the application ID, find the ID
- //by clicking on your application on Market home
- //page, and notice the ID from the address bar
Uninstall 應(yīng)用程序
- Uri uri = Uri.parse("file:///sdcard/song.mp3");
- Intent it = new Intent(Intent.ACTION_VIEW, uri);
- it.setType("audio/mp3");
- startActivity(it);
- Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, "1");
- Intent it = new Intent(Intent.ACTION_VIEW, uri);
- startActivity(it);
- Market 相關(guān)
- //尋找某個(gè)應(yīng)用
- Uri uri = Uri.parse("market://search?q=pname:pkg_name");
- Intent it = new Intent(Intent.ACTION_VIEW, uri); startActivity(it);
- //where pkg_name is the full package path for an application
- /顯示某個(gè)應(yīng)用的相關(guān)信息
- Uri uri = Uri.parse("market://details?id=app_id");
- Intent it = new Intent(Intent.ACTION_VIEW, uri); startActivity(it);
- /where app_id is the application ID, find the ID
- //by clicking on your application on Market home
- //page, and notice the ID from the address bar
【編輯推薦】