2017年7月6日 星期四

[轉]Android智能指針sp 和 wp 用法


Android手機操作系統既然是開源的操作系統。那麼在具體的文件夾中就會存放著各種相關功能的開源代碼。我們在使用的時候可以根據這些源代碼進行相應的修改就能輕鬆的完成我們所需的功能。在這里大家就一起來看看Android 智能指針的相關源碼解讀以及應用方法。
android的源代碼中,經常會看到形如:sp< xxx>、wp< xxx>這樣的類型定義,這其實是Android中的智能指針。智能指針是C++中的一個概念,通過基於引用計數的方法,解決對象的自動釋放的問題。在C++編程中,有兩個很讓人頭痛的問題:一是忘記釋放動態申請的對像從而造成內存洩露;二是對像在一個地方釋放後,又在別的地方被使用,從而引起內存訪問錯誤。
程序員往往需要花費很大精力進行精心設計,以避免這些問題的出現。在使用智能指針後,動態申請的內存將會被自動釋放(有點類似Java的垃圾回收),不需要再使用delete來釋放對象,也不需要考慮一個對像是否已經在其它地方被釋放了,從而使程序編寫工作減輕不少,而程序的穩定性大大提高。
Android智能指針相關的源代碼在下面兩個文件中:

frameworks\base\include\utils\RefBase.h
frameworks\base\libs\utils\RefBase.cpp
Android中定義了兩種智能指針類型,一種是強指針sp(strong pointer),一種是弱指針(weak pointer)。其實成為強引用和弱引用更合適一些。強指針與一般意義的智能指針概念相同,通過引用計數來記錄有多少使用者在使用一個對象,如果所有使用者都放棄了對該對象的引用,則該對象將被自動銷毀。
弱指針也指向一個對象,但是弱指針僅僅記錄該對象的地址,不能通過弱指針來訪問該對象,也就是說不能通過弱智真來調用對象的成員函數或訪問對象的成員變量。要想訪問弱指針所指向的對象,需首先將弱指針升級為強指針(通過wp類所提供的promote()方法)。弱指針所指向的對像是有可能在其它地方被銷毀的,如果對像已經被銷毀,wp的promote()方法將返回空指針,這樣就能避免出現地址訪問錯的情況。
是不是很神奇?弱指針是怎麼做到這一點的呢?其實說穿了一點也不復雜,原因就在於每一個可以被智能指針引用的對像都同時被附加了另外一個weakref_impl類型的對象,這個對像中負責記錄對象的強指針引用計數和弱指針引用計數。這個對像是Android智能指針的實現內部使用的,智能指針的使用者看不到這個對象。弱指針操作的就是這個對象,只有當強引用計數和弱引用計數都為0時,這個對象才會被銷毀。
說了這麼多原理,下面該看看到底智能指針該怎麼使用了。假設現在有一個類MyClass,如果要使用智能指針來引用這個類的對象,那麼這個類需滿足下列兩個前提條件:
(1) 這個類是基類RefBase的子類或間接子類;
(2) 這個類必須定義虛構造函數,即它的構造函數需要這樣定義:
  1. virtual ~MyClass(); 
滿足了上述條件的類就可以定義Android智能指針了,定義方法和普通指針類似。比如普通指針是這樣定義:
  1. MyClass* p_obj; 
Android智能指針是這樣定義:
  1. sp < MyClass >  p_obj; 
注意不要定義成sp< MyClass>* p_obj。初學者容易犯這種錯誤,這樣實際上相當於定義了一個指針的指針。儘管在語法上沒有問題,但是最好永遠不要使用這樣的定義。
定義了一個智能指針的變量,就可以像普通指針那樣使用它,包括賦值、訪問對象成員、作為函數的返回值、作為函數的參數等。比如:
  1. p_obj  =  new  MyClass();   
  2. //注意不要寫成  p_obj  =  new  sp < MyClass > 
  3. sp < MyClass > p_obj p_obj2  = p_obj;   
  4. p_obj- > func();  
  5. p_obj  =  create_obj ();  
  6. some_func(p_obj); 
注意不要試圖delete一個Android智能指針,即delete p_obj。不要擔心對象的銷毀問題,智能指針的最大作用就是自動銷毀不再使用的對象。不需要再使用一個對像後,直接將指針賦值為NULL即可:
  1. p_obj  =  NULL
上面說的都是強指針,弱指針的定義方法和強指針類似,但是不能通過弱指針來訪問對象的成員。下面是弱指針的示例:
  1. wp < MyClass > wp_obj  =  new  MyClass();   
  2. p_obj  =  wp_obj .promote();   
  3. //升級為強指針。不過這裡要用.而不是- > ,真是有負其指針之名啊  
  4. wp_obj  =  NULL
Android智能指針用起來是很方便,在一般情況下最好使用智能指針來代替普通指針。但是需要知道一個智能指針其實是一個對象,而不是一個真正的指針,因此其運行效率是遠遠比不上普通指針的。所以在對運行效率敏感的地方,最好還是不要使用智能指針為好

2016年6月27日 星期一

Android中padding與layout_margin的區別與用法

   林炳文Evankaka原創作品。轉載請注明出處http://blog.csdn.net/evankaka

 一、定義   

        android:layout_margin就是設置view的上下左右邊框的額外空間
        android:padding是設置內容相對view的邊框的距離
        padding,含義為“填充”,像墊肩壓類似的填充物,一個控制項的padding及此控制項內部的填充,由此可見padding是以所被定義的控制項Aparent控制項,而內部的內容物與控制項A的間距。而layout_marginA控制項所在的控制項為parent控制項,是A與其的間距。
        其實概念很簡單,padding是站在父view的角度描述問題,它規定它裡面的內容必須與這個父view邊界的距離。margin則是站在自己的角度描述問題,規定自己和其他(上下左右)的view之間的距離,如果同一級只有一個view,那麼它的效果基本上就和padding一樣了
http://img.blog.csdn.net/20150303131301841?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvRXZhbmtha2E=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center
[html] view plain copy
 在CODE上查看代码片派生到我的代码片
1.  當按鈕分別設置以上兩個屬性時,得到的效果是不一樣的。  
2.  android:paddingLeft="30px"  
3.  按鈕上設置的內容(例如圖片)離按鈕左邊邊界30個圖元  
4.  android:layout_marginLeft="30px"  
5.  整個按鈕離左邊設置的內容30個圖元  



二、使用示例

1、兩個都不加
[html] view plain copy
 在CODE上查看代码片派生到我的代码片
1.  xml version="1.0" encoding="utf-8"?>  
2.  <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
3.      android:layout_width="match_parent"  
4.      android:layout_height="match_parent"  
5.      android:background="#ffffff" >  
6.    
7.      <TextView  
8.          android:layout_width="100dip"  
9.          android:layout_height="100dip"  
10.          android:background="#000000"  
11.         android:text="Hello" />  
12.   
13. </RelativeLayout>  
效果:
http://img.blog.csdn.net/20150303132257741?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvRXZhbmtha2E=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center

2、只加padding
[html] view plain copy
 在CODE上查看代码片派生到我的代码片
1.  xml version="1.0" encoding="utf-8"?>  
2.  <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
3.      android:layout_width="match_parent"  
4.      android:layout_height="match_parent"  
5.       android:padding ="100dip"  
6.      android:background="#ffffff" >  
7.    
8.      <TextView  
9.          android:layout_width="100dip"  
10.         android:layout_height="100dip"  
11.          android:background="#000000"  
12.         android:text="Hello" />  
13.   
14. </RelativeLayout>  
效果:
http://img.blog.csdn.net/20150303132407088?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvRXZhbmtha2E=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center
3、只加margin
[html] view plain copy
 在CODE上查看代码片派生到我的代码片
1.  xml version="1.0" encoding="utf-8"?>  
2.  <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
3.      android:layout_width="match_parent"  
4.      android:layout_height="match_parent"  
5.      android:background="#ffffff" >  
6.    
7.      <TextView  
8.          android:layout_width="100dip"  
9.          android:layout_height="100dip"  
10.          android:background="#000000"  
11.          android:layout_margin="30dip"  
12.         android:text="Hello" />  
13.   
14. </RelativeLayout>  
效果:
http://img.blog.csdn.net/20150303132606979?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvRXZhbmtha2E=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center

三、注意說明

        LinearLayoutRelativeLayoutTableLayout中,這2個屬性都是設置都是有效的
        FrameLayout中,android:layout_margin是無效的,因為FrameLayout裡面的元素都是從左上角開始繪製的
        AbsoluteLayout中,沒有android:layout_margin屬性
        在此提醒,xml參數中含有layout的參數專案為定義的控制項相對于parent的關聯,沒有的一般為本身的定義,以上內容與此相符。又類似於gravitylayout_gravity,layout的是相對於parent的大體位置,而不帶的是自身內部內容的大體位置。

   林炳文Evankaka原創作品。轉載請注明出處http://blog.csdn.net/evankaka