Android智能指針使用方法介紹
Android手機操作系統(tǒng)既然是開源的操作系統(tǒng)。那么在具體的文件夾中就會存放著各種相關(guān)功能的開源代碼。我們在使用的時候可以根據(jù)這些源代碼進行相應(yīng)的修改就能輕松的完成我們所需的功能。在這里大家就一起來看看Android智能指針的相關(guān)源碼解讀以及應(yīng)用方法。#t#
在Android的源代碼中,經(jīng)常會看到形如:sp< xxx>、wp< xxx>這樣的類型定義,這其實是Android中的智能指針。智能指針是C++中的一個概念,通過基于引用計數(shù)的方法,解決對象的自動釋放的問題。在C++編程中,有兩個很讓人頭痛的問題:一是忘記釋放動態(tài)申請的對象從而造成內(nèi)存泄露;二是對象在一個地方釋放后,又在別的地方被使用,從而引起內(nèi)存訪問錯誤。
程序員往往需要花費很大精力進行精心設(shè)計,以避免這些問題的出現(xiàn)。在使用智能指針后,動態(tài)申請的內(nèi)存將會被自動釋放(有點類似Java的垃圾回收),不需要再使用delete來釋放對象,也不需要考慮一個對象是否已經(jīng)在其它地方被釋放了,從而使程序編寫工作減輕不少,而程序的穩(wěn)定性大大提高。
Android智能指針相關(guān)的源代碼在下面兩個文件中:
frameworks\base\include\utils\RefBase.h
frameworks\base\libs\utils\RefBase.cpp
Android中定義了兩種智能指針類型,一種是強指針sp(strong pointer),一種是弱指針(weak pointer)。其實成為強引用和弱引用更合適一些。強指針與一般意義的智能指針概念相同,通過引用計數(shù)來記錄有多少使用者在使用一個對象,如果所有使用者都放棄了對該對象的引用,則該對象將被自動銷毀。
弱指針也指向一個對象,但是弱指針僅僅記錄該對象的地址,不能通過弱指針來訪問該對象,也就是說不能通過弱智真來調(diào)用對象的成員函數(shù)或訪問對象的成員變量。要想訪問弱指針所指向的對象,需首先將弱指針升級為強指針(通過wp類所提供的promote()方法)。弱指針所指向的對象是有可能在其它地方被銷毀的,如果對象已經(jīng)被銷毀,wp的promote()方法將返回空指針,這樣就能避免出現(xiàn)地址訪問錯的情況。
是不是很神奇?弱指針是怎么做到這一點的呢?其實說穿了一點也不復(fù)雜,原因就在于每一個可以被智能指針引用的對象都同時被附加了另外一個 weakref_impl類型的對象,這個對象中負責(zé)記錄對象的強指針引用計數(shù)和弱指針引用計數(shù)。這個對象是Android智能指針的實現(xiàn)內(nèi)部使用的,智能指針的使用者看不到這個對象。弱指針操作的就是這個對象,只有當強引用計數(shù)和弱引用計數(shù)都為0時,這個對象才會被銷毀。
說了這么多原理,下面該看看到底智能指針該怎么使用了。假設(shè)現(xiàn)在有一個類MyClass,如果要使用智能指針來引用這個類的對象,那么這個類需滿足下列兩個前提條件:
(1) 這個類是基類RefBase的子類或間接子類;
(2) 這個類必須定義虛構(gòu)造函數(shù),即它的構(gòu)造函數(shù)需要這樣定義:
- virtual ~MyClass();
滿足了上述條件的類就可以定義Android智能指針了,定義方法和普通指針類似。比如普通指針是這樣定義:
- MyClass* p_obj;
Android智能指針是這樣定義:
- sp< MyClass> p_obj;
注意不要定義成 sp< MyClass>* p_obj。初學(xué)者容易犯這種錯誤,這樣實際上相當于定義了一個指針的指針。盡管在語法上沒有問題,但是最好永遠不要使用這樣的定義。
定義了一個智能指針的變量,就可以象普通指針那樣使用它,包括賦值、訪問對象成員、作為函數(shù)的返回值、作為函數(shù)的參數(shù)等。比如:
- p_obj = new MyClass();
- // 注意不要寫成 p_obj = new sp< MyClass>
- sp< MyClass> p_objp_obj2 = p_obj;
- p_obj->func();
- p_obj = create_obj();
- some_func(p_obj);
注意不要試圖delete一個Android智能指針,即 delete p_obj。不要擔(dān)心對象的銷毀問題,智能指針的最大作用就是自動銷毀不再使用的對象。不需要再使用一個對象后,直接將指針賦值為NULL即可:
- p_obj = NULL;
上面說的都是強指針,弱指針的定義方法和強指針類似,但是不能通過弱指針來訪問對象的成員。下面是弱指針的示例:
- wp< MyClass> wp_obj = new MyClass();
- p_obj = wp_obj.promote();
- // 升級為強指針。不過這里要用.而不是->,真是有負其指針之名啊
- wp_obj = NULL;
Android智能指針用起來是很方便,在一般情況下最好使用智能指針來代替普通指針。但是需要知道一個智能指針其實是一個對象,而不是一個真正的指針,因此其運行效率是遠遠比不上普通指針的。所以在對運行效率敏感的地方,最好還是不要使用智能指針為好。