深入解讀Android開(kāi)發(fā)中Activity的生命周期
什么是Activity
首先,Activity是Android系統(tǒng)中的四大組件之一,可以用于顯示View。Activity是一個(gè)與用記交互的系統(tǒng)模塊,幾乎所有的Activity都是和用戶(hù)進(jìn)行交互的,但是如果這樣就能說(shuō)Activity主要是用來(lái)顯示View就不太正確了。
在深入了解Activity之前,我們先要知道一下MVC設(shè)計(jì)模式,在JAVAEE 中MVC設(shè)計(jì)模式已經(jīng)很經(jīng)典了,而且分的也比較清晰了,但是在Android中,好多人對(duì)MVC在Android開(kāi)發(fā)中的應(yīng)用不是很清楚,下面我就先來(lái)介紹一下MVC在Android開(kāi)發(fā)中的應(yīng)用:
M(Model 模型):Model是應(yīng)用程序的主體部分,所有的業(yè)務(wù)邏輯都應(yīng)該寫(xiě)在這里,在Android中Model層與JavaEE中的變化不大,如:對(duì)數(shù)據(jù)庫(kù)的操作,對(duì)網(wǎng)絡(luò)等的操作都放在該層(但不是說(shuō)它們都放在同一個(gè)包中,可以分開(kāi)放,但它們統(tǒng)稱(chēng)為Model層)。
V(View 視圖):是應(yīng)用程序中負(fù)責(zé)生成用戶(hù)界面的部分,也是在整個(gè)MVC架構(gòu)中用戶(hù)唯一可以看到的一層,接收用戶(hù)輸入,顯示處理結(jié)果;在Android應(yīng)用中一般采用XML文件里德界面的描述,使用的時(shí)候可以非常方便的引入,當(dāng)然也可以使用JavaScript+Html等方式作為View。
C(Controller控制層)android的控制層的重任就要落在眾多的activity的肩上了,所以在這里就要建議大家不要在activity中寫(xiě)太多的代碼,盡量能過(guò)activity交割Model業(yè)務(wù)邏輯層處理。
好了,在介紹過(guò)Android應(yīng)用開(kāi)發(fā)中的MVC架構(gòu)后,我們就可以很明確的知道,在Android中Activity主要是用來(lái)做控制的,它可以選擇要顯示的View,也可以從View中獲取數(shù)據(jù)然后把數(shù)據(jù)傳給Model層進(jìn)行處理,最后再來(lái)顯示出處理結(jié)果。
Activity生命周期
接下來(lái)我們就一起來(lái)看一下Activity的生命周期,為什么要看這個(gè)呢?我想學(xué)過(guò)servlet的童鞋都知道,我們當(dāng)時(shí)在學(xué)習(xí)servlet的時(shí)候也是從生命周期開(kāi)始的,首先知道請(qǐng)求的順序,然后知道怎么處理請(qǐng)求的。只有這樣我們才能在需要的時(shí)候做一些相關(guān)的事情。
android的學(xué)習(xí)也是這樣,Activity是android中最重要的,一般情況下都表示一個(gè)界面,這樣,我們?nèi)绻枰M(jìn)行什么處理都需要在activity上進(jìn)行處理。下面我們來(lái)看一下具體的代碼。
1)先建一個(gè)項(xiàng)目吧,名稱(chēng)就隨便啦。Activity也隨便,反正自己看得懂就OK了。
我建了一個(gè)結(jié)構(gòu)如圖:
這里的名稱(chēng)是隨便的,不必追求一致。
2)大家看到我有兩個(gè)Activity,前面也講了Activity是一個(gè)界面,那么兩個(gè)當(dāng)然就是兩個(gè)界面啦。
看代碼之前,我們先來(lái)看看Activity事件的方法,有如下幾個(gè):
public void onCreate(Bundle savedInstanceState) ; public void onDestroy(); public void onPause(); public void onRestart(); public void onResume(); public void onStart(); public void onStop();
這些方法,我們看方法名大概就可以猜到那個(gè)方法是在什么時(shí)候調(diào)用的了。但真正要理解卻不是那么簡(jiǎn)單的。
下面開(kāi)始我們的正題,看代碼啦。
首先看一下strings.xml,這里定義了我們需要顯示的字符串。
<?xml version="1.0" encoding="utf-8"?> <resources> <string name="hello">Hello World, MainActivity!</string> <string name="app_name">Activity生命周期</string> <string name="other">另外一個(gè)Activity</string> <string name="start">啟動(dòng)另一個(gè)Activity</string> <string name="stop">結(jié)束當(dāng)前Activity</string> </resources>
前面兩個(gè)可有可無(wú)了,只是創(chuàng)建后沒(méi)有進(jìn)行修改。
我們來(lái)看一下界面的XML:
<?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>
這里大家應(yīng)該大概都可以看得懂啦,非常簡(jiǎn)單的,就兩個(gè)按鈕而已,一個(gè)是開(kāi)始另一個(gè)Activity,另一個(gè)是停止當(dāng)前的Activity。非常簡(jiǎn)單,如果這里看不懂,就需要先補(bǔ)充一下android的基本知識(shí)啦。
還有一個(gè)是androidManifest.xml這個(gè)是最主要的,因?yàn)槲覀冇昧藘蓚€(gè)Activity,兩個(gè)都需要在這里進(jìn)行“注冊(cè)”,否則將不可使用。
<?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的順序跟你啟動(dòng)的順序有關(guān),我這里需要以MainActivity來(lái)啟動(dòng),所以放在OtherActivity前面。
3)這里我們看完了界面代碼,接下來(lái)便是我們最主要的代碼啦。MainActivity和OtherActivity。我們通過(guò)MainActivity來(lái)進(jìn)行啟動(dòng)。點(diǎn)擊開(kāi)始另一個(gè)Activity則啟動(dòng)OtherActivity。停止則相應(yīng)的停止當(dāng)前的Activity。
下面是MainActivity的代碼:
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代碼如下:
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(); } }
代碼沒(méi)啥好講的羅,很簡(jiǎn)單的。主要還是要理解生命周期。
下面我們看一下運(yùn)行的效果:
當(dāng)我們進(jìn)行啟動(dòng)的時(shí)候:
它首先啟動(dòng)onCreate,其次是onStart,再來(lái)就是onResume。這三個(gè)特定,不用懷疑。
界面如下:
這時(shí)我們點(diǎn)擊啟動(dòng)另一個(gè)Activity看看效果,這里是MainActivity,點(diǎn)擊后將會(huì)跳轉(zhuǎn)到OtherActivity。
我們看看效果:
這里已經(jīng)跳轉(zhuǎn)到另一個(gè)Activity,因?yàn)槲覀冇昧送瑯拥膌ayout,所以顯示的是一樣的,只是標(biāo)題不一樣。
我們來(lái)看看它的信息:
我們看到MainActivity會(huì)首先暫停,然后再進(jìn)行新的Activity的創(chuàng)建過(guò)程,創(chuàng)建完成后就調(diào)用銷(xiāo)毀MainActivity。
大家估計(jì)會(huì)懷疑,因?yàn)槲以贛ainActivity中把finish寫(xiě)在startActivity后面,肯定是因?yàn)檫@個(gè)才在后面調(diào)用stop的。首先不是肯定的這么快,我們來(lái)把finish放到前面來(lái)看看。
我們把MainActivity中的onCreate中的部分代碼修改成下面的:
start.setOnClickListener(new View.OnClickListener() { public void onClick(View v) { MainActivity.this.finish(); Intent intent = new Intent(MainActivity.this,OtherActivity.class); startActivity(intent); } });
這里我們把finish放到前面了,我們重新運(yùn)行后可以看到效果:
我們看到效果是一樣的。但為了代碼的易讀,建議還是放到后面,符合生命周期的調(diào)用順序。
現(xiàn)在我們?cè)贠therActivity中,我們想要回到MainActivity中,還是同樣啟動(dòng)另一個(gè)Activity:
這里的調(diào)用過(guò)程也是一樣,首先暫停,創(chuàng)建,最后銷(xiāo)毀。
現(xiàn)在我們直接關(guān)閉MainActivity有:
這里我們看到它首先調(diào)用onPause方法,之后才是正式停止和銷(xiāo)毀。從這里我們猜想,當(dāng)發(fā)生一個(gè)Activity的切換時(shí),都會(huì)先調(diào)用當(dāng)前Activity的onPause方法。這里,我們先不說(shuō)正不正確,接下去看看,實(shí)踐證明一切。
4)我們?cè)谇懊媸鞘謩?dòng)finish了這個(gè)Activity,下面我們不結(jié)束了,我們看看修改的代碼:
OtherActivity中還是會(huì)調(diào)用finish來(lái)銷(xiāo)毀這個(gè)頁(yè)面,因?yàn)椴恍枰恕?br />
OtherActivity如下:
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); } });
這樣我們就改好了,其他地方并不需要修改。
我們看看效果,當(dāng)我們啟動(dòng)MainActivity后,我們點(diǎn)擊啟動(dòng)另一個(gè)Activity來(lái)啟動(dòng)OtherActivity,我們來(lái)看看信息:
我們看到它并沒(méi)有調(diào)用onDestroy,這表明這個(gè)onDestroy是當(dāng)我們調(diào)用finish方法才會(huì)調(diào)用的。
下面我們?cè)贠therActivity中點(diǎn)擊啟動(dòng)另一個(gè)Activity來(lái)啟動(dòng)另一個(gè)Activity,注意,這里的“啟動(dòng)”并不是真正啟動(dòng),不是通過(guò)startActivity來(lái)啟動(dòng)的,只是結(jié)束當(dāng)前的Activity。
我們看看信息:
這里我們?cè)冱c(diǎn)擊啟動(dòng)另一個(gè)Activity,實(shí)際上只是停止當(dāng)前的OtherActivity。我們看看打印出來(lái)的信息:
這里我們看到它并不是調(diào)用MainActivity的onCreate方法,因?yàn)镸ainActivity并沒(méi)有被銷(xiāo)毀,而只是停止了而已。當(dāng)OtherActivity被銷(xiāo)毀,要切換的時(shí)候,會(huì)重新啟動(dòng)MainActivity。這時(shí)就會(huì)調(diào)用onRestart。
5)注意,我們這里以把OtherActivity作為一個(gè)整的Activity,即是占整個(gè)界面的。下面我們來(lái)看另外一種情況,即我們啟動(dòng)的是一個(gè)對(duì)話(huà)框。
我們需要修改androidManifest.xml,修改后如下:
<activity android:name=".OtherActivity" android:label="@string/other" android:theme="@android:style/Theme.Dialog"> </activity>
只需要修改OtherActivity的定義即可,這里我們把它定義成Dialog即對(duì)話(huà)框類(lèi)型,它并不會(huì)占據(jù)整個(gè)界面,而原來(lái)的Activity后退到后臺(tái)運(yùn)行。
為了效果的演示,我們先把MainActivity中啟動(dòng)另一個(gè)Activity中的finish代碼去掉,而把OtherActivity中的啟動(dòng)另一個(gè)Activity的代碼去掉,留下finish代碼。這樣就可以看到重新啟動(dòng)的效果。
去掉后的代碼如下:
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); } });
OtherActivity中的代碼:
Button start = (Button)findViewById(R.id.start); start.setOnClickListener(new View.OnClickListener() { public void onClick(View v) { OtherActivity.this.finish(); } });
我們運(yùn)行后點(diǎn)擊啟動(dòng)另一個(gè)Activity可以看到:
這里我們看到MainActivity只調(diào)用了一個(gè)onPause,也就是說(shuō)它并沒(méi)有停止,只是在后臺(tái)運(yùn)行。
我們看效果:
我們看到主的Activity跑到后面并形成半透明的狀態(tài),這是它正在后臺(tái)運(yùn)行。
當(dāng)我們點(diǎn)擊啟動(dòng)另一個(gè)Activity,注意,我們這里并不啟動(dòng)另一個(gè)Activity,而只是直接結(jié)束當(dāng)前的Activity。
現(xiàn)在當(dāng)我們點(diǎn)擊啟動(dòng)另一個(gè)Activity時(shí),當(dāng)前的OtherActivity會(huì)關(guān)閉,而回到原來(lái)的MainActivity。
我們來(lái)看看效果:
而后臺(tái)的信息如下:
它直接調(diào)用onResume重新恢復(fù)MainActivity,而不是重新啟動(dòng)。這是因?yàn)镸ainActivity還在后臺(tái)運(yùn)行,我們不需要再重新啟動(dòng)一個(gè)。
其實(shí)這個(gè)生命周期并不難,難的是每個(gè)階段要理解。隨著接觸的東西越來(lái)越多,相信對(duì)這個(gè)周期的理解會(huì)越來(lái)越深的。
下面給一個(gè)android官方的生命周期的圖大家一起學(xué)習(xí):
相關(guān)文章
Android TextView實(shí)現(xiàn)跑馬燈效果的方法
這篇文章主要介紹了Android TextView跑馬燈效果實(shí)現(xiàn)方法,涉及Android布局文件中相關(guān)屬性的設(shè)置技巧,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-01-01Android實(shí)現(xiàn)截屏方式整理(總結(jié))
本篇文章主要介紹了Android 截屏方式整理(總結(jié)),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-07-07Android獲取RecyclerView滑動(dòng)距離方法詳細(xì)講解
RecyclerView是Android一個(gè)更強(qiáng)大的控件,其不僅可以實(shí)現(xiàn)和ListView同樣的效果,還有優(yōu)化了ListView中的各種不足。其可以實(shí)現(xiàn)數(shù)據(jù)縱向滾動(dòng),也可以實(shí)現(xiàn)橫向滾動(dòng)(ListView做不到橫向滾動(dòng))。接下來(lái)講解RecyclerView的用法2023-01-01Android中View自定義組合控件的基本編寫(xiě)方法
這篇文章主要介紹了Android中View自定義組合控件的基本編寫(xiě)方法,可以在布局的時(shí)候更加隨意地繼承,需要的朋友可以參考下2016-04-04android計(jì)算器實(shí)現(xiàn)兩位數(shù)的加減乘除
這篇文章主要為大家詳細(xì)介紹了android計(jì)算器實(shí)現(xiàn)兩位數(shù)的加減乘除,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-03-03Android studio實(shí)現(xiàn)滑動(dòng)開(kāi)關(guān)
這篇文章主要為大家詳細(xì)介紹了Android studio實(shí)現(xiàn)滑動(dòng)開(kāi)關(guān),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-03-03