在Android系統的江湖中有四大元件:活動(Activity), 服務(Service), 廣播接收器(Broadcast Reciver)和内容提供者(Content Provider)。今天所介紹的就是Android開發中的四大元件之一:Activity,其他那三大元件以後再進行介紹。說道Android中的Activity,如果你做過iOS開發的話,Activity類似于iOS中的ViewController(視圖控制器)。在應用中能看到的東西都是放在活動中的。活動是安卓開發比較重要的東西,是使用者互動和資料的入口。本篇部落格要介紹的内容是活動的建立,活動的跳轉與值的透傳。
iOS中的ViewController也是有自己的生命周期的,了解Activity或者ViewController的生命周期是很有必要的,這要你才能搞明白在什麼時間做什麼事情,關于iOS開發的東西請詳看之前的部落格:《我的iOS開發系列博文》和《我的Objective-C系列文章》。好廢話少說,Activity将要登場。
使用Android Studio建立一個Android的Add New Activity工程(步驟略,詳見上篇部落格),在這個工程中預設會有一個Blank Activity,而且在Blank Activity中自動添加了一個Text View, 上面寫着"Hello World", 運作這個工程就會看到Hello World在一個白色的活動中。
我們把這個工程中建立的一個新的Activity,然後再有Hello World中添加一個按鈕,點選按鈕跳轉到我們建立的新的Activity中。在新的按鈕中點選Back傳回到上一個Activity中。
一、Activity的建立與元件添加
1.建立一個Blank Activity
在Android Studio的檔案資源目錄中,選中你要建立Activity的包,右鍵單擊->New -> Activity ->各種Activity, 在這兒我們選擇Blank Activity,點選即可,操作如下圖所示。
點選完後出現下面的對話框來建立一個Activity,也就是我們自定義的Activity。Activity Name: 活動的名字,Layout Name: 活動對應布局檔案的名字,Titile: 上面導覽列顯示的名字。Menu Resource Name: 菜單的xml配置檔案的名稱(稍後會詳細介紹),點選Finishi即可。
活動建立後會在資源目錄中生成三個檔案,如下圖所示,java中的SecondActivity檔案有前面的"C"标志可知,是Java的Class檔案,也就是Activity對應的源檔案。而Layout檔案中的activity_second.xml是Activity對應的布局檔案,在布局檔案中你可以指定給Activity添加那些控件,并且可以控制這些控件的樣式和位置。第三個就是menu下的menu_second.xml檔案,該檔案是定義導航中的下拉菜單内容的,稍後給大家示範。
2. 控件添加
往Activity中添加控件就需要操作我們的Layout檔案夾下Activity所對應的xml檔案了。接下來要做的事情是在MainActivity中的布局檔案中添加一個按鈕,然後在代碼中擷取一個按鈕,并且監聽按鈕的點選事件,按鈕點選事件觸發後彈出一個提示框。詳細的看一下如何去添加控件并監聽控件的事件。
(1) 添加按鈕
打開activity_main.xml布局檔案, 切換到Design模式下,在Design模式下你可以以拖拽的方式來建立控件,以及定位控件。下方是添加了一個Button, 并且Button上的Text為ShowToast(Toast是安卓中顯示資訊用的元件)。
你也可以切換到Text模式下去看xml的文本,下面的大框中就是我們剛才拖拽出來的Button所對應的xml的内容,其中包括與布局有關的資訊:控件的寬高,上下左右的邊距等,還有控件的一些屬性:控件獨有的id以及控件顯示的文本(Text)等。當然如果對xml布局的屬性較為了解,就可以使用純代碼去聲明和布局你的控件。
(2)在代碼中擷取控件
經過上面的步驟,已經聲明并配置好了一個button。如果想再代碼中進行控件的使用,首先得通過上面button的id來執行個體化控件。下面的代碼是在MainActivity檔案中的onCreat()方法中添加的,關于onCreate方法,稍後的Activity的生命周期會詳細的介紹到。在Java源碼中可以通過findViewById來執行個體化Layout布局檔案中指定的控件。上面button的id為“myFirstButton”,是以執行個體化該button的方法如下:
//擷取界面上的按鈕
Button myFirstButton = (Button) findViewById(R.id.myFirstButton);
執行個體化Button後我們需要監聽按鈕的點選事件,下面使用的監聽方式類似iOS中的Block回調。就是給button賦一個點選按鈕執行的方法。點選按鈕就會執行下方的回調方法,并且可以通過Toast進行内容提示。
1 //按鈕點選的回調
2 myFirstButton.setOnClickListener(new View.OnClickListener() {
3 @Override
4 public void onClick(View v) {
5 Toast.makeText(MainActivity.this, "點選MyFirstButton", Toast.LENGTH_LONG).show();
6 }
7 });
經過上面的步驟,點選ShowToast按鈕就會提示“點選MyFirstButton”的内容,下方是執行的結果截圖:
二、Activity間的跳轉與傳值
在上面我們建立了一個名為SecondActivity的Blank Activity一直沒有,從上面的運作效果可以看出在MainActivity中有一個Go Second Activity的Button, 她就是用來跳轉到SecondActivity的,不同Activity見跳轉可以傳值,也可以在傳回的時候傳回值,接下來就介紹Activity間的跳轉和傳值問題。
1. 使用Intent進行Activity的跳轉
Intent在Android開發中被譽為“意圖”,從字面意思不難了解,就是“你打算去哪”。Intent與iOS開發中的NavigationController(導航控制器)極為相似,不過又有所不同。NavigationController是一個“棧”形式的容器,控制器可以通過push或者pop操作進行"入棧"和"出棧",這個入棧和出棧的操作也就是視圖控制器進行切換的操作。Intent實作原理也是一樣的,也有一個棧,這個棧中存儲的内容是一個個的Activity,Activity的Start和Finish操作對應着棧的Push和Pop操作。
(1)在"Go Second Activity" button的單擊事件中添加跳轉代碼,跳轉到SecondActiviy,代碼如下, 下方代碼是放在onCreate方法中。在下方使用的Intent類的構造器中,第一個參數是目前Activity,第二個參數是将要跳轉的Activity。通過startActivity方法來啟動Intent, 與其說是啟動Intent, 倒不如說事把目前Activity如棧,把将要顯示的Activity放入棧頂。
Button goSecondActivityButton = (Button) findViewById(R.id.go_sceond_button);
goSecondActivityButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//建立一個Intent(目前Activity, SecondActivity)=====顯示Intent
Intent intent = new Intent(MainActivity.this, SecondActivity.class);
//啟動Intent
startActivity(intent);
}
});
(2) 如果MainActivity中的值要傳給要跳轉過去的SecondActivity,那麼就可以通過Intent的對象中的putExtra方法來進行傳值。在上面的代碼startActivity()方法的上方加上下面這段代碼就是給目标Activity傳值。data是一個String類型的變量,其中存的值是要傳給将要跳轉的Activity的。對象intent通過調用putExtra來進行傳值,第一個參數是值得名字,也就是值的key, 在下個Activity中通過這個key來擷取對應的value。
1 //傳值給下一個Activity
2 String data = "我是上一個Activity中傳過來的值";
3
4 intent.putExtra("extra_data", data);
(3) 在新的Activity中我們需要擷取傳過來的值顯示在TextView上,并且點選一個Back按鈕傳回到上一層Activity,具體代碼如下。這些代碼是放在第二個Activity的onCreate()方法中的。可以通過getIntent擷取是通過那個Intent跳轉的,換句話說事擷取目前的導航棧。擷取Intent對象後,通過getStringExtra()方法傳入一個相應的鍵,通過這個鍵來擷取值。因為我們傳入的值是String類型的是以用getStringExtra(), 不同類型的值對應着不同的方法。然後把擷取的值顯示在TextView上。之後點選Button傳回。 Button中的finish()方法是結束目前Activity,就會自動傳回上一個Activity。TextView和Button也是通過拖拽的方式來擷取的,然後通過id進行事件的處理和指派。
1 //擷取上一個Activity傳過來的值
2 Intent intent = getIntent();
3 String data = intent.getStringExtra("extra_data");
4
5 //将擷取的值顯示在TextView上
6 TextView dataTextView = (TextView) findViewById(R.id.data_text_view);
7 dataTextView.setText(data);
8
9 //點選Button傳回上一個Activity
10 Button backButton = (Button) findViewById(R.id.bank_button);
11 backButton.setOnClickListener(new View.OnClickListener() {
12 @Override
13 public void onClick(View v) {
14 finish();
15 }
16 });
經過上面的步驟,最終運作效果如下,點選MainActivity中的Go Second Activity按鈕就會跳轉到第二個Activity,并且把第一個頁面中傳的值會在第二個Activity中進行顯示。點選SecondActivity中的Back按鈕就會執行finish()方法傳回上一個Activity。
2. 使用Intent打開系統功能
你可以以協定的方式打開系統的某些功能,比如打開系統的浏覽器,系統的撥号鍵盤等。在iOS也是這樣,不過是通過Application對象打開某些協定如tel://撥号協定等來調用系統功能。在安卓系統中也可以通過Intent對象來做這些操作。
(1)調用浏覽器打開連結的代碼如下,下方代碼是調用浏覽器打開連結。ACTION_VIEW是比較智能的,他會通過使用者傳入的資料來打開相應的應用,下方是通過setData傳入的網址,是以就會調用浏覽器,如果傳入的時tel:10010, 就會調用撥号盤。
Button openBaiduButton = (Button) findViewById(R.id.open_baidu_button);
openBaiduButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//調用本地浏覽器打開網址
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setData(Uri.parse("http://www.baidu.com"));
startActivity(intent);
}
});
(2)調用撥号盤你可以通過上面的方式改變一下傳入的資料即可,但是你還可以通過Intent.ACTION_DIAL也是可以調用撥号盤的,具體代碼如下所示:
1 Button telPhoneButton = (Button) findViewById(R.id.tel_number);
2 telPhoneButton.setOnClickListener(new View.OnClickListener() {
3 @Override
4 public void onClick(View v) {
5 Intent intent = new Intent(Intent.ACTION_DIAL);
6 intent.setData(Uri.parse("tel:10010"));
7 startActivity(intent);
8 }
9 });
點選上方兩個按鈕第一個會通過浏覽器打開網址,第二個會打開撥号盤,運作效果如下所示。
3.從傳回中的Activity中擷取值
從上面的執行個體中可知,我們可以把值從一個Activity中傳入到下一個Activity中。接下來要做的事情剛好相反,是從傳回的頁面中擷取值。這種傳值也是通過Intent來做的。我們在MainAcvitiy中添加一個按鈕“Go Third Activity”,點選按鈕跳轉到第三個Activity中,然後傳回擷取第三個Activity中傳過來的值。
(1)跳轉時通過startActivityForResult()方法來進行值得回傳,第一個參數就是intent對象,第二個參數是requestCode(請求碼)。requestCode在傳回後的回調方法中會使用到。
Button goThiredActivityButton = (Button) findViewById(R.id.go_third_button);
goThiredActivityButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//從下一個Activity中擷取資料
Intent intent = new Intent(MainActivity.this, ThirdActivity.class);
startActivityForResult(intent, 1);
}
});
(2)在ThirdActivity中要做的事情就是通過Intent傳回值,具體代碼如下所示,傳回值也是通過Intent對象的putExtra方法,然後去執行setReault方法即可。setResult()方法的第一個參數是resultCode(結果碼),也是在上一個Activity中接收值的回調中使用。
Button backButton = (Button) findViewById(R.id.third_back_button);
backButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = getIntent();
intent.putExtra("data_return", "我是第三個Activity中傳回的資料");
setResult(RESULT_OK, intent);
finish();
}
});
(3)接着就得在MainActivity中去重寫處理傳回值的回調方法了,也就是onActivityResult回調方法。在方法中通過key來擷取傳過來的值,并把值指派給MainActivity中的TextView具體代碼如下:
1 @Override
2 protected void onActivityResult(int requestCode, int resultCode, Intent data) {
3 switch (requestCode) {
4 case 1:
5 if (resultCode == RESULT_OK) {
6 String returnedData = data.getStringExtra("data_return");
7
8 TextView returnedDataTextView = (TextView) findViewById(R.id.return_textview);
9 returnedDataTextView.setText(returnedData);
10 }
11 break;
12 }
13 }
經過上面的步驟,運作效果如下,點選Go Third Activity會跳轉到ThirdActivity中,然後從ThirdActivity中點選Back按鈕進行傳回就會在上一個Activity中的TextView上顯示ThirdActivity中傳回的值,具體效果如下所示。
更多Android進階技術,面試資料系統整理分享,職業生涯規劃,産品,思維,行業觀察,談天說地。可以加Android架構師群;701740775。