原出处:http://blog.csdn.net/ydt_lwj/article/details/8131031

Android四大基本组件介绍与生命周期

Android四大基本组件介绍与生命周期

Android四大基本组件分别是Activity,Service服务,ContentProvider内容提供者,BroadcastReceiver广播接收器。

一:了解四大基本组件

Activity:

应用程序中,一个Activity通常就是一个单独的屏幕,它上面可以显示一些控件也可以监听并处理用户的事件做出响应。

Activity之间通过Intent进行通信。在Intent的描述结构中,有两个最重要的部分:动作和动作对应的数据。

典型的动作类型有:MAIN(activity的门户)、VIEW、PICK、EDIT等。而动作对应的数据则以URI的形式进行表示。例如:要查看一个人的联系方式,你需要创建一个动作类型为VIEW的intent,以及一个表示这个人的URI。

与之有关系的一个类叫IntentFilter。相对于intent是一个有效的做某事的请求,一个intentfilter则用于描述一个 activity(或者IntentReceiver)能够操作哪些intent。一个activity如果要显示一个人的联系方式时,需要声明一个 IntentFilter,这个IntentFilter要知道怎么去处理VIEW动作和表示一个人的URI。IntentFilter需要在 AndroidManifest.xml中定义。通过解析各种intent,从一个屏幕导航到另一个屏幕是很简单的。当向前导航时,activity将会调用startActivity(IntentmyIntent)方法。然后,系统会在所有安装的应用程序中定义的 IntentFilter中查找,找到最匹配myIntent的Intent对应的activity。新的activity接收到 myIntent的通知后,开始运行。当startActivity方法被调用将触发解析myIntent的动作,这个机制提供了两个关键好处:

A、Activities能够重复利用从其它组件中以Intent的形式产生的一个请求;

B、Activities可以在任何时候被一个具有相同IntentFilter的新的Activity取代。

AndroidManifest文件中含有如下过滤器的Activity组件为默认启动类当程序启动时系统自动调用它

<intent-filter><actionandroid:name="android.intent.action.MAIN"/><categoryandroid:name="android.intent.category.LAUNCHER"/></intent-filter>

BroadcastReceive广播接收器:

你的应用可以使用它对外部事件进行过滤只对感兴趣的外部事件(如当电话呼入时,或者数据网络可用时)进行接收并做出响应。广播接收器没有用户界面。然而,它们可以启动一个activity或serice来响应它们收到的信息,或者用NotificationManager来通知用户。通知可以用很多种方式来吸引用户的注意力──闪动背灯、震动、播放声音等。一般来说是在状态栏上放一个持久的图标,用户可以打开它并获取消息。

广播类型:

普通广播通过Context.sendBroadcast(IntentmyIntent)发送的

有序广播通过 Context.sendOrderedBroadcast(intent,receiverPermission)发送的,该方法第2个参数决定该广播的级别,级别数值是在-1000到1000之间,值越大,发送的优先级越高;广播接收者接收广播时的级别级别(可通过 intentfilter中的priority进行设置设为2147483647时优先级最高),同级别接收的先后是随机的,再到级别低的收到广播,高级别的或同级别先接收到广播的可以通过abortBroadcast()方法截断广播使其他的接收者无法收到该广播,还有其他构造函数

异步广播通过Context.sendStickyBroadcast(IntentmyIntent)发送的,还有 sendStickyOrderedBroadcast(intent,resultReceiver,scheduler,initialCode,initialData,initialExtras) 方法,该方法具有有序广播的特性也有异步广播的特性;发送异步广播要:<uses-permissionandroid:name="android.permission.BROADCAST_STICKY"/>权限,接收并处理完Intent后,广播依然存在,直到你调用removeStickyBroadcast(intent)主动把它去掉

注意:发送广播时的intent参数与Contex.startActivity()启动起来的Intent不同,前者可以被多个订阅它的广播接收器调用,后者只能被一个(Activity或service)调用

监听广播Intent步骤:

1>写一个继承BroadCastReceiver的类,重写onReceive()方法,广播接收器仅在它执行这个方法时处于活跃状态。当onReceive()返回后,它即为失活状态,注意:为了保证用户交互过程的流畅,一些费时的操作要放到线程里,如类名 SMSBroadcastReceiver

2>注册该广播接收者,注册有两种方法程序动态注册和AndroidManifest文件中进行静态注册(可理解为系统中注册)如下:

静态注册,注册的广播,下面的priority表示接收广播的级别"2147483647"为最高优先级

<receiverandroid:name=".SMSBroadcastReceiver">  <intent- filterandroid:priority="2147483647">    < actionandroid:name="android.provider.Telephony.SMS_RECEIVED"/>  < /intent-filter></receiver>

动态注册,一般在Activity可交互时onResume()内注册BroadcastReceiver

IntentFilterintentFilter=newIntentFilter("android.provider.Telephony.SMS_RECEIVED"); registerReceiver(mBatteryInfoReceiver,intentFilter);//反注册 unregisterReceiver(receiver);

注意:

1.生命周期只有十秒左右,如果在onReceive()内做超过十秒内的事情,就会报 ANR(ApplicationNoResponse)程序无响应的错误信息,如果需要完成一项比较耗时的工作,应该通过发送Intent给Service,由Service来完成.这里不能使用子线程来解决,因为BroadcastReceiver的生命周期很短,子线程可能还没有结束BroadcastReceiver就先结束了.BroadcastReceiver一旦结束,此时BroadcastReceiver的所在进程很容易在系统需要内存时被优先杀死,因为它属于空进程(没有任何活动组件的进程).如果它的宿主进程被杀死,那么正在工作的子线程也会被杀死.所以采用子线程来解决是不可靠的

2.动态注册广播接收器还有一个特点,就是当用来注册的Activity关掉后,广播也就失效了。静态注册无需担忧广播接收器是否被关闭,只要设备是开启状态,广播接收器也是打开着的。也就是说哪怕app本身未启动,该app订阅的广播在触发时也会对它起作用

系统常见广播Intent,如开机启动、电池电量变化、时间改变等广播

Service服务:

一个Service是一段长生命周期的,没有用户界面的程序,可以用来开发如监控类程序。

比较好的一个例子就是一个正在从播放列表中播放歌曲的媒体播放器。在一个媒体播放器的应用中,应该会有多个activity,让使用者可以选择歌曲并播放歌曲。然而,音乐重放这个功能并没有对应的activity,因为使用者当然会认为在导航到其它屏幕时音乐应该还在播放的。在这个例子中,媒体播放器这个activity会使用Context.startService()来启动一个service,从而可以在后台保持音乐的播放。同时,系统也将保持这个service一直执行,直到这个service运行结束。另外,我们还可以通过使用Context.bindService()方法,连接到一个service上(如果这个service还没有运行将启动它)。当连接到一个service之后,我们还可以service提供的接口与它进行通讯。拿媒体播放器这个例子来说,我们还可以进行暂停、重播等操作。

Service使用步骤如下

1>继承service类

2>AndroidManifast.xml配置清单文件中<application>节点里对服务进行配置

<servicename=".SMSService"/>

服务不能自己运行,需要通过Contex.startService()或Contex.bindService()启动服务

通过startService()方法启动的服务于调用者没有关系,即使调用者关闭了,服务仍然运行想停止服务要调用 Context.stopService(),此时系统会调用onDestory(),使用此方法启动时,服务首次启动系统先调用服务的 onCreate()-->onStart(),如果服务已经启动再次调用只会触发onStart()方法

使用bindService()启动的服务与调用者绑定,只要调用者关闭服务就终止,使用此方法启动时,服务首次启动系统先调用服务的 onCreate()-->onBind(),如果服务已经启动再次调用不会再触发这2个方法,调用者退出时系统会调用服务的 onUnbind()-->onDestory(),想主动解除绑定可使用Contex.unbindService(),系统依次调用 onUnbind()-->onDestory();

ContentProvider内容提供者:

android平台提供了ContentProvider使一个应用程序的指定数据集提供给其他应用程序。这些数据可以存储在文件系统中、在一个SQLite数据库、或以任何其他合理的方式,

其他应用可以通过ContentResolver类(见ContentProviderAccessApp例子)从该内容提供者中获取或存入数据.(相当于在应用外包了一层壳),

只有需要在多个应用程序间共享数据是才需要内容提供者。例如,通讯录数据被多个应用程序使用,且必须存储在一个内容提供者中

它的好处:统一数据访问方式。

android系统自带的内容提供者(顶级的表示数据库名,非顶级的都是表名)这些内容提供者在SDK文档的 android.providerJava包中都有介绍。见:http://developer.android.com/reference /android/provider/package-summary.html

├────Browser

├────CallLog

├────Contacts

│├────Groups

│├────People

│├────Phones

│└────Photos

├────Images

│└────Thumbnails

├────MediaStore

│├────Albums

│├────Artists

│├────Audio

│├────Genres

│└────Playlists

├────Settings

└────Video

CallLog:地址和接收到的电话信息

Contact.People.Phones:存储电话号码

Setting.System:系统设置和偏好设置

使用ContentProvider对外共享数据的步骤

1>继承ContentProvider类并根据需求重写以下方法:

    publicbooleanonCreate();//处理初始化操作/***插入数据到内容提供者(允许其他应用向你的应用中插入数据时重写)*@paramuri*@paraminitialValues插入的数据*@return*/publicUriinsert(Uriuri,ContentValuesinitialValues);/***从内容提供者中删除数据(允许其他应用删除你应用的数据时重写)*@paramuri*@paramselection条件语句*@paramselectionArgs参数*@return*/publicintdelete(Uriuri,Stringselection,String[]selectionArgs);/***更新内容提供者已存在的数据(允许其他应用更新你应用的数据时重写)*@paramuri*@paramvalues更新的数据*@paramselection条件语句*@paramselectionArgs参数*@return*/publicintupdate(Uriuri,ContentValuesvalues,Stringselection,String[]selectionArgs);/***返回数据给调用者(允许其他应用从你的应用中获取数据时重写)*@paramuri*@paramprojection列名*@paramselection条件语句*@paramselectionArgs参数*@paramsortOrder排序*@return*/publicCursorquery(Uriuri,String[]projection,Stringselection,String[]selectionArgs,StringsortOrder);/***用于返回当前Uri所代表数据的MIME类型*如果操作的数据为集合类型(多条数据),那么返回的类型字符串应该为 vnd.android.cursor.dir/开头*例如要得到所有person记录的Uri为content: //com.bravestarr.provider.personprovider/person,
    *   那么返回的MIME类型字符串应该为"vnd.android.cursor.dir/person"*如果操作的数据为单一数据,那么返回的类型字符串应该为vnd.android.cursor.item/开头*例如要得到id为10的person记录的Uri为 content://com.bravestarr.provider.personprovider/person/10,
    *   那么返回的MIME类型字符串应该为"vnd.android.cursor.item/person"*@paramuri*/publicStringgetType(Uriuri)

这些方法中的Uri参数,得到后需要进行解析然后做对应处理,Uri表示要操作的数据,包含两部分信息:

1.需要操作的contentprovider

2.对contentprovider中的什么数据进行操作,一个Uri格式:结构头://authorities(域名)/路径(要操作的数据,根据业务而定)

content://com.bravestarr.provider.personprovider/person/10

说明:contentprovider的结构头已经由android规定为content://

authorities用于唯一标识这个contentprovider程序,外部调用者可以根据这个找到他

路径表示我们要操作的数据,路径的构建根据业务而定.路径格式如下:

要操作person表行号为10的记录,可以这样构建/person/10

要操作person表的所有记录,可以这样构建/person

2>在AndroidManifest.xml中使用<provider>对ContentProvider进行配置注册(内容提供者注册它自己就像网站注册域名),ContentProvider采用authoritie(原意授权,可理解为域名)作为唯一标识,方便其他应用能找到

<applicationandroid:icon="@drawable/ic_launcher"android:label="@string /app_name"><!--authorities属性命名建议:公司名.provider.SomeProvider--><providerandroid:name=".PersonProvider"android:authorities="com.bravestarr.provider.personprovider"/>...</application>

关于四大基本组件的一个总结:

1>4大组件的注册

4大基本组件都需要注册才能使用,每个Activity、service、ContentProvider内容提供者都需要在 AndroidManifest文件中进行配置AndroidManifest文件中未进行声明的activity、服务以及内容提供者将不为系统所见,从而也就不可用,而BroadcastReceive广播接收者的注册分静态注册(在AndroidManifest文件中进行配置)和通过代码动态创建并以调用Context.registerReceiver()的方式注册至系统。需要注意的是在AndroidManifest文件中进行配置的广播接收者会随系统的启动而一直处于活跃状态,只要接收到感兴趣的广播就会触发(即使程序未运行)

AndroidManifest文件中进行注册格式如下:

<activity>元素的name属性指定了实现了这个activity的Activity的子类。icon和label属性指向了包含展示给用户的此activity的图标和标签的资源文件。

<service>元素用于声明服务

<receiver>元素用于声明广播接收器

<provider>元素用于声明内容提供者

2>4大组件的激活

•容提供者的激活:当接收到ContentResolver发出的请求后,内容提供者被激活。而其它三种组件──activity、服务和广播接收器被一种叫做intent的异步消息所激活

•Activity的激活通过传递一个Intent对象至Context.startActivity()或 Activity.startActivityForResult()以载入(或指定新工作给)一个activity。相应的activity可以通过调用getIntent()方法来查看激活它的intent。如果它期望它所启动的那个activity返回一个结果,它会以调用 startActivityForResult()来取代startActivity()。比如说,如果它启动了另外一个Activity以使用户挑选一张照片,它也许想知道哪张照片被选中了。结果将会被封装在一个Intent对象中,并传递给发出调用的activity的 onActivityResult()方法。

•服务的激活可以通过传递一个Intent对象至Context.startService()或Context.bindService() 前者Android调用服务的onStart()方法并将Intent对象传递给它,后者Android调用服务的onBind()方法将这个 Intent对象传递给它

•发送广播可以通过传递一个Intent对象至给Context.sendBroadcast()、

Context.sendOrderedBroadcast()或Context.sendStickyBroadcast()Android会调用所有对此广播有兴趣的广播接收器的onReceive()方法,将intent传递给它们

3>四大组件的关闭

内容提供者仅在响应ContentResolver提出请求的时候激活。而一个广播接收器仅在响应广播信息的时候激活。所以,没有必要去显式的关闭这些组件。

Activity关闭:可以通过调用它的finish()方法来关闭一个activity

服务关闭:对于通过startService()方法启动的服务要调用Context.stopService()方法关闭服务,使用bindService()方法启动的服务要调用Contex.unbindService()方法关闭服务

二:四大组件的生命周期

介绍生命周期之前,先提一下任务的概念

任务其实就是activity的栈它由一个或多个Activity组成的共同完成一个完整的用户体验,换句话说任务就是”应用程序”(可以是一个也可以是多个,比如假设你想让用户看到某个地方的街道地图。而已经存在一个具有此功能的activity了,那么你的activity所需要做的工作就是把请求信息放到一个Intent对象里面,并把它传递给startActivity()。于是地图浏览器就会显示那个地图。而当用户按下 BACK键的时候,你的activity又会再一次的显示在屏幕上,此时任务是由2个应用程序中的相关activity组成的)栈底的是启动整个任务的Activity,栈顶的是当前运行的用户可以交互的Activity,当一个activity启动另外一个的时候,新的activity就被压入栈,并成为当前运行的activity。而前一个activity仍保持在栈之中。当用户按下BACK键的时候,当前activity出栈,而前一个恢复为当前运行的activity。栈中保存的其实是对象,栈中的Activity永远不会重排,只会压入或弹出,所以如果发生了诸如需要多个地图浏览器的情况,就会使得一个任务中出现多个同一Activity子类的实例同时存在。

任务中的所有activity是作为一个整体进行移动的。整个的任务(即activity栈)可以移到前台,或退至后台。举个例子说,比如当前任务在栈中存有四个activity──三个在当前activity之下。当用户按下HOME键的时候,回到了应用程序加载器,然后选择了一个新的应用程序(也就是一个新任务)。则当前任务遁入后台,而新任务的根activity显示出来。然后,过了一小会儿,用户再次回到了应用程序加载器而又选择了前一个应用程序(上一个任务)。于是那个任务,带着它栈中所有的四个activity,再一次的到了前台。当用户按下BACK键的时候,屏幕不会显示出用户刚才离开的activity(上一个任务的根

activity)。取而代之,当前任务的栈中最上面的activity被弹出,而同一任务中的上一个activity显示了出来。

Activity栈:先进先出规则

Android四大基本组件介绍与生命周期

Android系统是一个多任务(Multi-Task)的操作系统,可以在用手机听音乐的同时,也执行其他多个程序。每多执行一个应用程序,就会多耗费一些系统内存,当同时执行的程序过多,或是关闭的程序没有正确释放掉内存,系统就会觉得越来越慢,甚至不稳定。

为了解决这个问题,Android引入了一个新的机制--生命周期(LifeCycle)。

Android应用程序的生命周期是由Android框架进行管理,而不是由应用程序直接控

制。通常,每一个应用程序(入口一般会是一个Activity的onCreate方法),都会产生

一个进程(Process)。当系统内存即将不足的时候,会依照优先级自动进行进程(process)的回收。不管是使用者或开发者,都无法确定的应用程序何时会被回收。所以为了很好的防止数据丢失和其他问题,了解生命周期很重要。

Activity生命周期:

Android四大基本组件介绍与生命周期

图3.1activity生命周期图

Activity整个生命周期的4种状态、7个重要方法和3个嵌套循环

1>四种状态

  1. 活动(Active/Running)状态

当Activity运行在屏幕前台(处于当前任务活动栈的最上面),此时它获取了焦点能响应用户的操作,属于运行状态,同一个时刻只会有一个Activity处于活动(Active)或运行

(Running)状态

  1. 暂停(Paused)状态

当Activity失去焦点但仍对用户可见(如在它之上有另一个透明的Activity或Toast、AlertDialog等弹出窗口时)它处于暂停状态。暂停的Activity仍然是存活状态(它保留着所有的状态和成员信息并保持和窗口管理器的连接),但是当系统内存极小时可以被系统杀掉

3.停止(Stopped)状态

完全被另一个Activity遮挡时处于停止状态,它仍然保留着所有的状态和成员信息。只是对用户不可见,当其他地方需要内存时它往往被系统杀掉

4.非活动(Dead)状态

Activity尚未被启动、已经被手动终止,或已经被系统回收时处于非活动的状态,要手动终止Activity,可以在程序中调用"finish"方法。

如果是(按根据内存不足时的回收规则)被系统回收,可能是因为内存不足了

内存不足时,Dalvak虚拟机会根据其内存回收规则来回收内存:

1.先回收与其他Activity或Service/IntentReceiver无关的进程(即优先回收独

立的Activity)因此建议,我们的一些(耗时)后台操作,最好是作成Service的形式

2.不可见(处于Stopped状态的)Activity

3.Service进程(除非真的没有内存可用时会被销毁)

4.非活动的可见的(Paused状态的)Activity

5.当前正在运行(Active/Running状态的)Activity

2>7个重要方法,当Activity从一种状态进入另一状态时系统会自动调用下面相应的方

法来通知用户这种变化

当Activity第一次被实例化的时候系统会调用,

整个生命周期只调用1次这个方法

通常用于初始化设置:1、为Activity设置所要使用的布局文件2、为按钮绑定监听器等静态的设置操作

onCreate(BundlesavedInstanceState);

当Activity可见未获得用户焦点不能交互时系统会调用

onStart();

当Activity已经停止然后重新被启动时系统会调用

onRestart();

当Activity可见且获得用户焦点能交互时系统会调用

onResume();

当系统启动另外一个新的Activity时,在新Activity启动之前被系统调用保存现有的Activity中的持久数据、停止动画等,这个实现方法必须非常快。当系统而不是用户自己出于回收内存时,关闭了activity之后。用户会期望当他再次回到这个activity的时候,它仍保持着上次离开时的样子。此时用到了onSaveInstanceState(),方法onSaveInstanceState()用来保存Activity 被杀之前的状态,在onPause()之前被触发,当系统为了节省内存销毁了Activity(用户本不想销毁)时就需要重写这个方法了,当此 Activity再次被实例化时会通过onCreate(BundlesavedInstanceState)将已经保存的临时状态数据传入因为 onSaveInstanceState()方法不总是被调用,触发条件为(按下HOME键,按下电源按键关闭屏幕,横竖屏切换情况下),你应该仅重写 onSaveInstanceState()来记录activity的临时状态,而不是持久的数据。应该使用onPause()来存储持久数据。

onPause();

当Activity被新的Activity完全覆盖不可见时被系统调用

onStop();

当Activity(用户调用finish()或系统由于内存不足)被系统销毁杀掉时系统调用,(整个生命周期只调用1次)用来释放onCreate()方法中创建的资源,如结束线程等

onDestroy();

3>3个嵌套循环

1.Activity完整的生命周期:从第一次调用onCreate()开始直到调用onDestroy()结束

2.Activity的可视生命周期:从调用onStart()到相应的调用onStop()

在这两个方法之间,可以保持显示Activity所需要的资源。如在onStart()中注册一个广播接收者监听影响你的UI的改变,在onStop()中注销。

3.Activity的前台生命周期:从调用onResume()到相应的调用onPause()。

举例说明:

例1:有3个Acitivity,分别用One,Two(透明的),Three表示,One是应用启动时的主Activity

启动第一个界面ActivityOne时,它的次序是

onCreate(ONE)-onStart(ONE)-onResume(ONE)

点"打开透明Activity"按钮时,这时走的次序是

onPause(ONE)-onCreate(TWO)-onStart(TWO)-onResume(TWO)

再点back回到第一个界面,Two会被杀这时走的次序是

onPause(TWO)-onActivityResult(ONE)-onResume(ONE)-onStop(TWO)-onDestroy(TWO)

点"打开全屏Activity"按钮时,这时走的次序是

onPause(ONE)-onCreate(Three)-onStart(Three)-onResume(Three)-onStop(ONE)

再点back回到第一个界面,Three会被杀这时走的次序是

onPause(Three)-onActivityResult(ONE)-onRestart(ONE)-onStart(ONE)-onResume(ONE)-onStop(Three)-onDestroy(Three)

再点back退出应用时,它的次序是

onPause(ONE)-onStop(ONE)-onDestroy(ONE)

例2:横竖屏切换时候Activity的生命周期

他切换时具体的生命周期是怎么样的:

1、新建一个Activity,并把各个生命周期打印出来

2、运行Activity,得到如下信息

onCreate-->
onStart-->
onResume-->

3、按crtl+f12切换成横屏时

onSaveInstanceState-->
onPause-->
onStop-->
onDestroy-->
onCreate-->
onStart-->
onRestoreInstanceState-->
onResume-->

4、再按crtl+f12切换成竖屏时,发现打印了两次相同的log

onSaveInstanceState-->
onPause-->
onStop-->
onDestroy-->
onCreate-->
onStart-->
onRestoreInstanceState-->
onResume-->
onSaveInstanceState-->
onPause-->
onStop-->
onDestroy-->
onCreate-->
onStart-->
onRestoreInstanceState-->
onResume-->

5、修改AndroidManifest.xml,把该Activity添加android:configChanges="orientation",执行步骤3

onSaveInstanceState-->
onPause-->
onStop-->
onDestroy-->
onCreate-->
onStart-->
onRestoreInstanceState-->
onResume-->

6、再执行步骤4,发现不会再打印相同信息,但多打印了一行onConfigChanged

onSaveInstanceState-->
onPause-->
onStop-->
onDestroy-->
onCreate-->
onStart-->
onRestoreInstanceState-->
onResume-->
onConfigurationChanged-->

7、把步骤5的android:configChanges="orientation"改成android:configChanges="orientation|keyboardHidden",执行步骤3,就只打印onConfigChanged

onConfigurationChanged-->

8、执行步骤4

onConfigurationChanged-->
onConfigurationChanged-->

总结:

1、不设置Activity的android:configChanges时,切屏会重新调用各个生命周期,切横屏时会执行一次,切竖屏时会执行两次

2、设置Activity的android:configChanges="orientation"时,切屏还是会重新调用各个生命周期,切横、竖屏时只会执行一次

3、设置Activity的android:configChanges="orientation|keyboardHidden"时,切屏不会重新调用各个生命周期,只会执行onConfigurationChanged方法


总结一下整个Activity的生命周期

补充一点,当前Activity产生事件弹出Toast和AlertDialog的时候Activity的生命周期不会有改变

Activity运行时按下HOME键(跟被完全覆盖是一样的):onSaveInstanceState-->onPause-->onStop,再次进入激活状态时:onRestart-->onStart--->onResume

BroadcastReceive广播接收器生命周期:

生命周期只有十秒左右,如果在onReceive()内做超过十秒内的事情,就会报ANR(ApplicationNoResponse)程序无响应的错误信息

它的生命周期为从回调onReceive()方法开始到该方法返回结果后结束

Service服务生命周期:

Android四大基本组件介绍与生命周期

图3.2service生命周期图

Service完整的生命周期:从调用onCreate()开始直到调用onDestroy()结束

Service有两种使用方法:

1>以调用Context.startService()启动,而以调用Context.stopService()结束

2>以调用Context.bindService()方法建立,以调用Context.unbindService()关闭

service重要的生命周期方法

当用户调用startService()或bindService()时,Service第一次被实例化的时候系统会调用,整个生命周期只调用1 次这个方法,通常用于初始化设置。注意:多次调用startService()或bindService()方法不会多次触发onCreate()方法

voidonCreate()

当用户调用stopService()或unbindService()来停止服务时被系统调用,(整个生命周期只调用1次)用来释放onCreate()方法中创建的资源

voidonDestroy()

通过startService()方法启动的服务

初始化结束后系统会调用该方法,用于处理传递给startService()的Intent对象。如音乐服务会打开Intent来探明将要播放哪首音乐,并开始播放。注意:多次调用startService()方法会多次触发onStart()方法

voidonStart(Intentintent)

通过bindService()方法启动的服务

初始化结束后系统会调用该方法,用来绑定传递给bindService的Intent的对象。注意:多次调用bindService()时,如果该服务已启动则不会再触发此方法

IBinderonBind(Intentintent)

用户调用unbindService()时系统调用此方法,Intent对象同样传递给该方法

booleanonUnbind(Intentintent)

如果有新的客户端连接至该服务,只有当旧的调用onUnbind()后,新的才会调用该方法

voidonRebind(Intentintent)

Activity:

activity是Android提供的界面,所有和用户交互的事情都发生在这个类。

Activity的生命周期:

OnCreate()创建的时候会调用这个方法设置这个Activity设置布局控件,初始化数据等。

OnStart()当控件被我们看到的时候,就调用OnStart()方法在OnCreate()之后或者OnStop()后调用。

OnResume()当重新获得用户焦点的时候就调用这个方法。

onPause()保护现场用的可能被另外一个透明的Dialog窗口覆盖,失去焦点,但是他仍然和窗口管理器保持连接,系统可以继续保护Activity的内部状态。

onStop()停止和OnStart()对应,注意OnStop()和OnPause()不一样的地方是他是完全被另外一个窗口覆灭。也就是失去焦点而且不可见。

onDestory()销毁

创建一个Activity的要点

1.一个Activity就是一个类,并且这个类要继承Activity

2.需要复写onCreate方法

3.要在Androidmanifest.xml进行配置(四大组件都要进行配置)

4.为Activity添加必要的控件

在一个Activity启动另外一个Activity

Intentintent=newIntent(CurrentActivity.this,OtherActivity.class)//OtherActivity也要在AndroidManifest.xml中注册。

startActivity(intent);

Intent还可以放入各种数据传入下一个Activity。

可以用下面的方式接收传过来的数据:

还可以用Bundel的方法发送数据。Bundel就像一个承载数据的List<>里面可以放各种数据。但是他的key只能是string类型的,放入Intent的方式也是intent.putExtra(bundle);

Service:

Service在什么时候会用到呢?比如说音乐程序的时候,需要后台运行,但是又看不到界面,此时就用到Service。这也是Service和Activity最大的不同。

Service分类:

本地Service和远程Service。

本地Service是同一个进程内Service彼此之间共同的内存区域。

远程Service是同一个系统内的不同进程之间访问(注意是同一个系统内,不是不同系统)。

Service生命周期:

onCreate()->onStart()->onDestory()三个方法继承的时候如果要用到记得重新写下。

创建一个Service要点:

1.创建一个类,要继承Service

2.要在androidManifest.xml中进行配置

3.在Activity中配置调用它

BroadcastReceiver:

在android中可以通过broadcast告诉其他程序发生了什么事情。比如电源,比如短信,比如信号。

BroadcastReceiver相比于其他组件比较特别的地方是他还提供了一种所谓的热注册,就是不用写在androidManifest.xml中,直接用

registerReceiver(Broadcastreceiver,IntentFilter)注册。然后用unregisterReceiver()消除注册。

一些常用的Action:

ACTION_CALLactivity启动一个电话.
ACTION_EDITactivity显示用户编辑的数据.
ACTION_MAINactivity作为Task中第一个

Activity启动
ACTION_SYNCactivity同步手机与数据服务器上的数据.
ACTION_BATTERY_LOWbroadcastreceiver电池电量过低警告.
ACTION_HEADSET_PLUGbroadcastreceiver插拔耳机警告
ACTION_SCREEN_ONbroadcastreceiver屏幕变亮警告.
ACTION_TIMEZONE_CHANGEDbroadcastreceiver改变时区警告.

ContentProvider:

这个组件比较复杂

ContentProvider可以支持多个应用的数据共享。可以用来操作音频,视频,图片,私人通讯录等等。但是要记得获取适当的读取权限。当然也可以用来公开自己的数据。(难道ContentProvider是全局变量?整个手机的程序都能访问得到?)

创建ContentProvider要点:

1.创建一个ContentProvider首先要继承ContentProvider类。

2.在androidManifest中配置

3.在Activity类中调用insert()query()可以调用

实际上所有的contentprovider用户都不能直接访问contentprovider实例,只能通过ContentResolver的中间代理。而要得到ContentResolver的实例可以用Activity的getContentResolver方法。如下

可以自己去调用ContentProvider相应的方法。

Uri使用方法:来自http://blog.sina.com.cn/s/blog_5688414b0100xagp.html

为系统的每一个资源给其一个名字,比方说通话记录。

1、每一个ContentProvider都拥有一个公共的URI,这个URI用于表示这个ContentProvider所提供的数据。

2、Android所提供的ContentProvider都存放在android.provider包中。将其分为A,B,C,D4个部分:

A:标准前缀,用来说明一个ContentProvider控制这些数据,无法改变的;"content://"

B:URI的标识,它定义了是哪个ContentProvider提供这些数据。对于第三方应用程序,为了保证URI标识的唯一性,它必须是一个完整的、小写的类名。这个标识在元素的authorities属性中说明:一般是定义该ContentProvider的包.类的名称;"content://hx.android.text.myprovider"

C:路径,不知道是不是路径,通俗的讲就是你要操作的数据库中表的名字,或者你也可以自己定义,记得在使用的时候保持一致就ok了

"content://hx.android.text.myprovider/tablename"

D:如果URI中包含表示需要获取的记录的ID;则就返回该id对应的数据,如果没有ID,就表示返回全部;"content://hx.android.text.myprovider/tablename/#"#表示数据id

数据模型:

ContentProvider返回的数据结构是Cursor类似于DataSet,每一个contentprovider定义一个唯一公开的 URI,用于指定他的数据集。一个contentprovider可以包含多个数据集。这样就需要多个URI与之对应。URI的开头必须是“content://”

表示这个Uri指定一个contentprovider。

增删查改:

查询

Cursorcursor=getContentResolver().query(

ContactsContract.CommonDataKinds.Phone.CONTENT_URI,null,null,null,null);

得到查询数据:

Cursor.moveToNext()读取到下一条类似于C#的sdr.read()

条件发在哪里?

可以放在

修改:

插入:

数据放在values内,values是键值对。如下:

ContentValuesvalues=newContentValues();

values.put(People.NAME,"AbrahamLincoln");

Uriuri=getContentResolver().insert(People.CONTENT_URI,values);

更多相关文章

  1. android复制数据库到SD卡(网上搜集,未经验证)
  2. 解决EditText不显示光标的三种方法(总结)
  3. JS判断Android、iOS或浏览器的多种方法(四种方法)
  4. Android中SQLite数据库操作(1)——使用SQL语句操作SQLite数据库
  5. Android 数据存储
  6. Android View onMeasure 方法
  7. Android Studio 3.0以后打包修改文件名方法

随机推荐

  1. Android常见错误之[email&#160;protected
  2. Android Root原理分析及防Root新思路
  3. android拨打电话流程分析
  4. Android获取Apk文件图标信息
  5. Android内核和驱动的详细介绍
  6. Android材料设计之材料主题
  7. 巧用Android图片资源,打造更精致的APP
  8. Android自定义视图三:给自定义视图添加“
  9. 【原创】Android 耗电信息统计服务——Ba
  10. android mms流播放器