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());
- // }
- }
本文详细介绍了如何使用Parcelable接口封装和传递复杂对象数据,包括实现步骤、数据绑定、数据获取方法及Parcel内存机制的理解。同时,展示了如何在不同Activity间通过Intent传递封装的对象,并解析获取对象内容。
886

被折叠的 条评论
为什么被折叠?



