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());
- // }
- }