什么是activity
首先,activity是android系統中的四大組件之一,可以用于顯示view。activity是一個與用記交互的系統模塊,幾乎所有的activity都是和用戶進行交互的,但是如果這樣就能說activity主要是用來顯示view就不太正確了。
在深入了解activity之前,我們先要知道一下mvc設計模式,在javaee 中mvc設計模式已經很經典了,而且分的也比較清晰了,但是在android中,好多人對mvc在android開發中的應用不是很清楚,下面我就先來介紹一下mvc在android開發中的應用:
m(model 模型):model是應用程序的主體部分,所有的業務邏輯都應該寫在這里,在android中model層與javaee中的變化不大,如:對數據庫的操作,對網絡等的操作都放在該層(但不是說它們都放在同一個包中,可以分開放,但它們統稱為model層)。
v(view 視圖):是應用程序中負責生成用戶界面的部分,也是在整個mvc架構中用戶唯一可以看到的一層,接收用戶輸入,顯示處理結果;在android應用中一般采用xml文件里德界面的描述,使用的時候可以非常方便的引入,當然也可以使用javascript+html等方式作為view。
c(controller控制層)android的控制層的重任就要落在眾多的activity的肩上了,所以在這里就要建議大家不要在activity中寫太多的代碼,盡量能過activity交割model業務邏輯層處理。
好了,在介紹過android應用開發中的mvc架構后,我們就可以很明確的知道,在android中activity主要是用來做控制的,它可以選擇要顯示的view,也可以從view中獲取數據然后把數據傳給model層進行處理,最后再來顯示出處理結果。
activity生命周期
接下來我們就一起來看一下activity的生命周期,為什么要看這個呢?我想學過servlet的童鞋都知道,我們當時在學習servlet的時候也是從生命周期開始的,首先知道請求的順序,然后知道怎么處理請求的。只有這樣我們才能在需要的時候做一些相關的事情。
android的學習也是這樣,activity是android中最重要的,一般情況下都表示一個界面,這樣,我們如果需要進行什么處理都需要在activity上進行處理。下面我們來看一下具體的代碼。
1)先建一個項目吧,名稱就隨便啦。activity也隨便,反正自己看得懂就ok了。
我建了一個結構如圖:
這里的名稱是隨便的,不必追求一致。
2)大家看到我有兩個activity,前面也講了activity是一個界面,那么兩個當然就是兩個界面啦。
看代碼之前,我們先來看看activity事件的方法,有如下幾個:
1
2
3
4
5
6
7
|
public void oncreate(bundle savedinstancestate) ; public void ondestroy(); public void onpause(); public void onrestart(); public void onresume(); public void onstart(); public void onstop(); |
這些方法,我們看方法名大概就可以猜到那個方法是在什么時候調用的了。但真正要理解卻不是那么簡單的。
下面開始我們的正題,看代碼啦。
首先看一下strings.xml,這里定義了我們需要顯示的字符串。
1
2
3
4
5
6
7
8
|
<?xml version= "1.0" encoding= "utf-8" ?> <resources> <string name= "hello" >hello world, mainactivity!</string> <string name= "app_name" >activity生命周期</string> <string name= "other" >另外一個activity</string> <string name= "start" >啟動另一個activity</string> <string name= "stop" >結束當前activity</string> </resources> |
前面兩個可有可無了,只是創建后沒有進行修改。
我們來看一下界面的xml:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
<?xml version= "1.0" encoding= "utf-8" ?> <linearlayout xmlns:android= "http://schemas.android.com/apk/res/android" android:orientation= "vertical" android:layout_width= "fill_parent" android:layout_height= "fill_parent" > <button android:layout_width= "wrap_content" android:layout_height= "wrap_content" android:text= "@string/start" android:id= "@+id/start" /> <button android:layout_width= "wrap_content" android:layout_height= "wrap_content" android:text= "@string/stop" android:id= "@+id/stop" /> </linearlayout> |
這里大家應該大概都可以看得懂啦,非常簡單的,就兩個按鈕而已,一個是開始另一個activity,另一個是停止當前的activity。非常簡單,如果這里看不懂,就需要先補充一下android的基本知識啦。
還有一個是androidmanifest.xml這個是最主要的,因為我們用了兩個activity,兩個都需要在這里進行“注冊”,否則將不可使用。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
<?xml version= "1.0" encoding= "utf-8" ?> <manifest xmlns:android= "http://schemas.android.com/apk/res/android" package = "com.shun.android" android:versioncode= "1" android:versionname= "1.0" > <uses-sdk android:minsdkversion= "7" /> <application android:icon= "@drawable/icon" android:label= "@string/app_name" > <activity android:name= ".mainactivity" 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= ".otheractivity" android:label= "@string/other" > <intent-filter> <action android:name= "android.intent.action.main" /> <category android:name= "android.intent.category.launcher" /> </intent-filter> </activity> </application> </manifest> |
注意,這里的activity的順序跟你啟動的順序有關,我這里需要以mainactivity來啟動,所以放在otheractivity前面。
3)這里我們看完了界面代碼,接下來便是我們最主要的代碼啦。mainactivity和otheractivity。我們通過mainactivity來進行啟動。點擊開始另一個activity則啟動otheractivity。停止則相應的停止當前的activity。
下面是mainactivity的代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
|
package com.shun.android; import android.app.activity; import android.content.intent; import android.os.bundle; import android.util.log; import android.view.view; import android.widget.button; public class mainactivity extends activity { private static final string tag = "mainactivity" ; @override public void oncreate(bundle savedinstancestate) { super .oncreate(savedinstancestate); setcontentview(r.layout.main); log.i(tag, "oncreate" ); button start = (button)findviewbyid(r.id.start); start.setonclicklistener( new view.onclicklistener() { public void onclick(view v) { intent intent = new intent(mainactivity. this ,otheractivity. class ); startactivity(intent); mainactivity. this .finish(); } }); button stop = (button)findviewbyid(r.id.stop); stop.setonclicklistener( new view.onclicklistener() { public void onclick(view v) { mainactivity. this .finish(); } }); } protected void ondestroy(){ log.i(tag, "ondestroy" ); super .ondestroy(); } protected void onpause(){ log.i(tag, "onpause" ); super .onpause(); } protected void onrestart(){ log.i(tag, "onrestart" ); super .onrestart(); } protected void onresume(){ log.i(tag, "onresume" ); super .onresume(); } protected void onstart(){ log.i(tag, "onstart" ); super .onstart(); } protected void onstop(){ log.i(tag, "onstop" ); super .onstop(); } } |
otheractivity代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
|
package com.shun.android; import android.app.activity; import android.content.intent; import android.os.bundle; import android.util.log; import android.view.view; import android.widget.button; public class otheractivity extends activity{ private static final string tag = "otheractivity" ; @override public void oncreate(bundle savedinstancestate) { super .oncreate(savedinstancestate); setcontentview(r.layout.main); log.i(tag, "oncreate" ); button start = (button)findviewbyid(r.id.start); start.setonclicklistener( new view.onclicklistener() { public void onclick(view v) { intent intent = new intent(otheractivity. this ,mainactivity. class ); startactivity(intent); otheractivity. this .finish(); } }); button stop = (button)findviewbyid(r.id.stop); stop.setonclicklistener( new view.onclicklistener() { public void onclick(view v) { otheractivity. this .finish(); } }); } protected void ondestroy(){ log.i(tag, "ondestroy" ); super .ondestroy(); } protected void onpause(){ log.i(tag, "onpause" ); super .onpause(); } protected void onrestart(){ log.i(tag, "onrestart" ); super .onrestart(); } protected void onresume(){ log.i(tag, "onresume" ); super .onresume(); } protected void onstart(){ log.i(tag, "onstart" ); super .onstart(); } protected void onstop(){ log.i(tag, "onstop" ); super .onstop(); } } |
代碼沒啥好講的羅,很簡單的。主要還是要理解生命周期。
下面我們看一下運行的效果:
當我們進行啟動的時候:
它首先啟動oncreate,其次是onstart,再來就是onresume。這三個特定,不用懷疑。
界面如下:
這時我們點擊啟動另一個activity看看效果,這里是mainactivity,點擊后將會跳轉到otheractivity。
我們看看效果:
這里已經跳轉到另一個activity,因為我們用了同樣的layout,所以顯示的是一樣的,只是標題不一樣。
我們來看看它的信息:
我們看到mainactivity會首先暫停,然后再進行新的activity的創建過程,創建完成后就調用銷毀mainactivity。
大家估計會懷疑,因為我在mainactivity中把finish寫在startactivity后面,肯定是因為這個才在后面調用stop的。首先不是肯定的這么快,我們來把finish放到前面來看看。
我們把mainactivity中的oncreate中的部分代碼修改成下面的:
1
2
3
4
5
6
7
8
|
start.setonclicklistener( new view.onclicklistener() { public void onclick(view v) { mainactivity. this .finish(); intent intent = new intent(mainactivity. this ,otheractivity. class ); startactivity(intent); } }); |
這里我們把finish放到前面了,我們重新運行后可以看到效果:
我們看到效果是一樣的。但為了代碼的易讀,建議還是放到后面,符合生命周期的調用順序。
現在我們在otheractivity中,我們想要回到mainactivity中,還是同樣啟動另一個activity:
這里的調用過程也是一樣,首先暫停,創建,最后銷毀。
現在我們直接關閉mainactivity有:
這里我們看到它首先調用onpause方法,之后才是正式停止和銷毀。從這里我們猜想,當發生一個activity的切換時,都會先調用當前activity的onpause方法。這里,我們先不說正不正確,接下去看看,實踐證明一切。
4)我們在前面是手動finish了這個activity,下面我們不結束了,我們看看修改的代碼:
otheractivity中還是會調用finish來銷毀這個頁面,因為不需要了。
otheractivity如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
button start = (button)findviewbyid(r.id.start); start.setonclicklistener( new view.onclicklistener() { public void onclick(view v) { otheractivity. this .finish(); } }); mainactivity如下: button start = (button)findviewbyid(r.id.start); start.setonclicklistener( new view.onclicklistener() { public void onclick(view v) { intent intent = new intent(mainactivity. this ,otheractivity. class ); startactivity(intent); } }); |
這樣我們就改好了,其他地方并不需要修改。
我們看看效果,當我們啟動mainactivity后,我們點擊啟動另一個activity來啟動otheractivity,我們來看看信息:
我們看到它并沒有調用ondestroy,這表明這個ondestroy是當我們調用finish方法才會調用的。
下面我們在otheractivity中點擊啟動另一個activity來啟動另一個activity,注意,這里的“啟動”并不是真正啟動,不是通過startactivity來啟動的,只是結束當前的activity。
我們看看信息:
這里我們再點擊啟動另一個activity,實際上只是停止當前的otheractivity。我們看看打印出來的信息:
這里我們看到它并不是調用mainactivity的oncreate方法,因為mainactivity并沒有被銷毀,而只是停止了而已。當otheractivity被銷毀,要切換的時候,會重新啟動mainactivity。這時就會調用onrestart。
5)注意,我們這里以把otheractivity作為一個整的activity,即是占整個界面的。下面我們來看另外一種情況,即我們啟動的是一個對話框。
我們需要修改androidmanifest.xml,修改后如下:
1
2
3
4
|
<activity android:name= ".otheractivity" android:label= "@string/other" android:theme= "@android:style/theme.dialog" > </activity> |
只需要修改otheractivity的定義即可,這里我們把它定義成dialog即對話框類型,它并不會占據整個界面,而原來的activity后退到后臺運行。
為了效果的演示,我們先把mainactivity中啟動另一個activity中的finish代碼去掉,而把otheractivity中的啟動另一個activity的代碼去掉,留下finish代碼。這樣就可以看到重新啟動的效果。
去掉后的代碼如下:
mainactivity中的部分代碼:
1
2
3
4
5
6
7
8
|
button start = (button)findviewbyid(r.id.start); start.setonclicklistener( new view.onclicklistener() { public void onclick(view v) { intent intent = new intent(mainactivity. this ,otheractivity. class ); startactivity(intent); } }); |
otheractivity中的代碼:
1
2
3
4
5
6
7
|
button start = (button)findviewbyid(r.id.start); start.setonclicklistener( new view.onclicklistener() { public void onclick(view v) { otheractivity. this .finish(); } }); |
我們運行后點擊啟動另一個activity可以看到:
這里我們看到mainactivity只調用了一個onpause,也就是說它并沒有停止,只是在后臺運行。
我們看效果:
我們看到主的activity跑到后面并形成半透明的狀態,這是它正在后臺運行。
當我們點擊啟動另一個activity,注意,我們這里并不啟動另一個activity,而只是直接結束當前的activity。
現在當我們點擊啟動另一個activity時,當前的otheractivity會關閉,而回到原來的mainactivity。
我們來看看效果:
而后臺的信息如下:
它直接調用onresume重新恢復mainactivity,而不是重新啟動。這是因為mainactivity還在后臺運行,我們不需要再重新啟動一個。
其實這個生命周期并不難,難的是每個階段要理解。隨著接觸的東西越來越多,相信對這個周期的理解會越來越深的。
下面給一個android官方的生命周期的圖大家一起學習: