Parcelable:一種封裝數據的容器。
實現的方法:
1)writeToParcel方法。
2)describeContents方法。
3 ) public static final Parcelable.Creator<Person> CREATOR = new Parcelable.Creator<Person>() {
《1》先了解Parcelable的簡單用法
我們在activity之間進行數據傳遞時,傳遞一些簡單的基本類型,android是提供寫好的方法的。但我們
需要傳遞一個實體類,比如一個stu對象,就沒有相關的api了,這個時候我們可以使用Parcelable.
package com.example.test0001.domain; import java.util.HashMap; import android.os.Parcel; import android.os.Parcelable; public class Person implements Parcelable { public HashMap<String, String> map = new HashMap<String, String>(); public String name; @Override public int describeContents() { return 0; } @Override public void writeToParcel(Parcel dest, int flags) { dest.writeMap(map); dest.writeString(name); } public static final Parcelable.Creator<Person> CREATOR = new Parcelable.Creator<Person>() { @SuppressWarnings("unchecked") @Override public Person createFromParcel(Parcel source) { Person p = new Person(); p.map = source.readHashMap(HashMap.class.getClassLoader()); p.name = source.readString(); return p; } @Override public Person[] newArray(int size) { return null; } }; }
以上是對person對象的封裝,將對象封裝到包裹中。
《2》下面就簡單的傳遞數據,先了解兩個常用的數據綁定和數據獲取的方法。
================================發送數據============================================ package com.example.test0001; import java.util.HashMap; import com.example.test0001.domain.Person; import android.app.Activity; import android.content.Intent; import android.os.Bundle; //發送的Activity public class TestNew extends Activity { @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Intent intent = new Intent(); Person p = new Person(); p.map = new HashMap<String,String>(); p.map.put("yes", "ido"); p.name="ok"; intent.putExtra("yes", p); intent.setClass(this, Test.class); startActivity(intent); } }
================================接收數據============================================ package com.example.test0001; import com.example.test0001.domain.Person; import android.app.Activity; import android.content.Intent; import android.os.Bundle; public class Test extends Activity { @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.test); Intent i = getIntent(); Person p = i.getParcelableExtra("yes"); System.out.println("---->"+p.name); System.out.println("---->"+p.map.size()); } }
《3》下面是比較難的部分,關於Parcel分析
Parcel的內存機制:
C語言中結構體的內存對齊和Parcel採用的內存存放機制一樣 ,即讀取最小字節爲32bit,也是
4個字節。高於4個字節的,以實際數據類型進行存放,但必須爲4byte的倍數;
實際存放字節:
實際讀取字節:
另外一個注意點就是我們在writeXXX()和readXXX()時,導致的偏移量是共用的,例如,我們在writeInt(23)後,
此時的datapostion=4,如果我們想讀取5,簡單的通過readInt()是不行的,只能得到0。這時我們只能通過
setDataPosition(0)設置爲起始偏移量,從起始位置讀取四個字節,即23。因此,在讀取某個值時,可能需要使用
setDataPostion(int postion)使偏移量裝換到我們的值處。
巧用setDataPosition()方法,當我們的parcel對象中只存在某一類型時,我們就可以通過這個方法來快速的讀取所有值。具體方法如下:
- /**
- * 前提條件,Parcel存在多個類型相同的對象,本例子以10個float對象說明:
- */
- public void readSameType() {
- Parcel parcel =Parcel.obtain() ;
- for (int i = 0; i < 10; i++) {
- parcel.writeDouble(i);
- Log.i(TAG, "write double ----> " + getParcelInfo());
- }
- //方法一 ,顯示設置偏移量
- int i = 0;
- int datasize = parcel.dataSize();
- while (i < datasize) {
- parcel.setDataPosition(i);
- double fvalue = parcel.readDouble();
- Log.i(TAG, " read double is=" + fvalue + ", --->" + getParcelInfo());
- i += 8; // double佔用字節爲 8byte
- }
- // 方法二,由於對象的類型一致,我們可以直接利用readXXX()讀取值會產生偏移量
- // parcel.setDataPosition(0) ; //
- // while(parcel.dataPosition()<parcel.dataSize()){
- // double fvalue = parcel.readDouble();
- // Log.i(TAG, " read double is=" + fvalue + ", --->" + getParcelInfo());
- // }
- }