对象的序列化存储:Serializable 和 Parceable

简介:

在进行Android开发的时候我们有时候需要用到数据的持久化存储,或者在进程之间传递数据。其中就可能需要用到对象的序列化,经过序列化的对象之后可以通过Intent或者Boundle来传输了。接下来还是想些介绍下吧。

1.什么叫序列化,什么叫反序列化

序列化: 将数据结构或对象转换成二进制串的过程。反序列化:将在序列化过程中所生成的二进制串转换成数据结构或者对象的过程。

简单来说,序列化就是将我们生成的对象进行存储起来(比如磁盘上),以用来将来使用或者在网络上进行传输,而反序列化呢,就是由我们的之前序列化生成的二进制串重新生成对象的过程。注意,这里我们反复说的序列化啦,反序列化啦,都是针对的对象,而非类。因为我们是针对对象进行存取与传输的,而非类,当我们需要重新获取之前的对象的时候,是直接读取出来的(从文件或网络中),而非根据类new出一个对象,这点是需要注意的。

2.如何序列化

序列话的方式有两种,一种是实现Serializable接口,一种是实现Parceable接口,下面会具体介绍这两种方式。

a.实现Serializable接口

这种序列化方式是Java提供的,它的优点是简单,其实Serializable接口是个空接口,因而我们并不需要实现什么抽象方法,但是我们却往往需要在类中声明一个静态变量标识(serialVersionUID),但这不是必须的,我们不声明,依然可以实现序列化,但是这样的话会对反序列化产生一定的影响,可能会在我们对类做了修改之后而造成对象的反序列化失败。声明方式如下:

 
 
  1. private static final long serialVersionUID = 8711368828010083044L; 

注意,这里的值可以是任意值。

下面我们来具体实现下。

 
 
  1. package com.qc.admin.myserializableparceabledemo; 
  2.  
  3. import java.io.Serializable
  4.  
  5. /** 
  6.  * Created by admin on 2016/12/1. 
  7.  */ 
  8.  
  9. public class User implements Serializable { 
  10.  
  11.     private static final long serialVersionUID = 519067123721295773L; 
  12.  
  13.     public int userId; 
  14.     public String userName; 
  15.     public boolean isMale; 
  16.  
  17.     public User(int userId, String userName, boolean isMale) { 
  18.  
  19.         this.userId = userId; 
  20.         this.userName = userName; 
  21.         this.isMale = isMale; 
  22.  
  23.     } 
  24.  
  25.     @Override 
  26.     public String toString() { 
  27.         return "User{ " + 
  28.                 "userId = " + userId + 
  29.                 ", userName = " + userName + 
  30.                 ", isMale = " + isMale + 
  31.                 " }"
  32.     } 
  33.  

下面是序列化与反序列化过程:

 
 
  1. private void beginSerizable() throws IOException, ClassNotFoundException { 
  2.  
  3.         // 序列化 
  4.         User user = new User(2016, "qian"true); 
  5.         ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(new File(getFilesDir(), "myfile.txt"))); 
  6.         out.writeObject(user); 
  7.         out.close(); 
  8.  
  9.         // 反序列化 
  10.         // 注意,这里后面的“/myfile.txt”前面有个斜杠“/”,否则会报“FileNotFoundException”异常 
  11.         ObjectInputStream in = new ObjectInputStream(new FileInputStream(getFilesDir() + "/myfile.txt")); 
  12.         User mUser = (Userin.readObject(); 
  13.         textView.setText(mUser.toString()); 
  14.         in.close(); 
  15.         Log.i("test",mUser.toString()); 
  16.     }  

运行结果截图:

注意:如果是在Android项目中调用以上方法,别忘了在Manifest.xml文件中配置如下权限:

 
 
  1. <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> 
  2. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>  

b.实现Parceable接口

这种方式是Android提供的方式,相比较前面那种方式来讲,这种方式稍微有点复杂,我们需要自己尽享序列化与反序列化的操作,但是它却更加高效,并不需要执行大量的I/O操作。而且这种方式也是Android推荐的序列化方式,因此我们应该首选Parceable。只要实现了这个接口,一个类的对象就可以实现序列化并可以通过Intent和Binder进行传递了。下面请看实例:

 
 
  1. public class Book implements Parcelable { 
  2.     public String bookTitle; 
  3.     public int bookId; 
  4.  
  5.     protected Book(Parcel in) { 
  6.         bookTitle = in.readString(); 
  7.         bookId = in.readInt(); 
  8.     } 
  9.  
  10.     public static final Creator<Book> CREATOR = new Creator<Book>() { 
  11.         @Override 
  12.         public Book createFromParcel(Parcel in) { 
  13.             return new Book(in); 
  14.         } 
  15.  
  16.         @Override 
  17.         public Book[] newArray(int size) { 
  18.             return new Book[size]; 
  19.         } 
  20.     }; 
  21.  
  22.     @Override 
  23.     public int describeContents() { 
  24.         return 0; 
  25.     } 
  26.  
  27.     @Override 
  28.     public void writeToParcel(Parcel parcel, int i) { 
  29.         parcel.writeString(bookTitle); 
  30.         parcel.writeInt(bookId); 
  31.     } 
  32.  

这里将Book这个类就实现了Parcelable接口,其实在Android Studio IDE中,上述过程很简单,我们只需要定义一个类,实现Parcelable接口,然后在里面定义我们的属性或者说是字段,根据提示的错误,按照它提示的方法覆盖相应的方法,之后的一切其实都可以自动生成(不过如果需要构造方法的话,那就需要自动生成了,toString()方法也是自己实现的),所以不用担心在Android开发中通过实现Parceable接口会比较麻烦,因为AS都会为你自动生成。上面我们已经完整的将Book类实现了Parceable接口,那接下来如何序列化和反序列化呢?如果你说,刚才不是已经说过了吗,采用文件读取的方式不久可以了啦...当你那样做的时候,你会发现会报如下的错误:

Why???...什么情况?提示我们Book类没有实现序列化:

 
 
  1. /System.err: java.io.NotSerializableException: com.qc.admin.myserializableparceabledemo.Book 

好啦,之所以出现这种问题,并不是我们的实现过程有问题,而是使用该类的方式行不通。到这里我们就明白了Serializable和Parceable两种方式实现序列化还是有区别的,刚才我们也讲了,Parceable更加高效,不会像Serializable那样有大量的I/O操作,这句话的具体含义就道出了Serializable与Parcelable区别:虽然两者都是用于支持序列化、反序列化话操作,但是两者最大的区别在于存储媒介的不同,Serializable是将序列化后的对象存储在硬盘上,使用I/O读写的方式,而Parcelable是将其存储在内存中,是针对内存的读写,熟悉计算机组成原理的朋友都知道,内存的读写速度显然要远远大于I/O的读写速度,这也是为什么Android中推荐使用Parcelable这种方式来实现对象的序列化。

那我们应该怎么使用通过实现Parcelable接口实现序列化的对象呢?答案是:通过Intent方式传递,除了基本类型外,Intent只能传输序列化之后的对象,对应这两种序列化方式,也有两种相应的方法:

 
 
  1. mIntent.getSerializableExtra(string name );  
 
 
  1. mIntent.getParcelableExtra(String name ); 

当然,放入的操作就没有这种区分了,都是方法:

 
 
  1. mIntent.putExtra(); 

我们可以在第一个Activity中将序列化对象放入Intent,在另一个Activity中取出,比如:在另一端获取对象,例如:

 
 
  1. Bundle mBundle = getIntent().getExtras(); 
  2. Book mBook = mBundle.getParcelable("book1");  

下面再看类User实现Parceable接口的过程,它内部包含了一个可序列化的类Book,具体细节跟上面的有点不同:

 
 
  1. package com.qc.admin.myserializableparceabledemo; 
  2.  
  3. import android.os.Parcel; 
  4. import android.os.Parcelable; 
  5.  
  6. /** 
  7.  * Created by admin on 2016/12/1. 
  8.  */ 
  9.  
  10. public class User implements Parcelable { 
  11.  
  12.     public int userId; 
  13.     public String userName; 
  14.     public boolean isMale; 
  15.     public Book book; 
  16.  
  17.  
  18.     public User(int userId, String userName, boolean isMale, Book book) { 
  19.  
  20.         this.userId = userId; 
  21.         this.userName = userName; 
  22.         this.isMale = isMale; 
  23.         this.book = book; 
  24.     } 
  25.  
  26.  
  27.     protected User(Parcel in) { 
  28.         userId = in.readInt(); 
  29.         userName = in.readString(); 
  30.         isMale = in.readByte() != 0; 
  31.         // 此为不同之处1 
  32.         // 也可以通过这种方式:book = in.readParcelable(Thread.currentThread().getContextClassLoader()); 
  33.         book = in.readParcelable(Book.class.getClassLoader()); 
  34.          
  35.     } 
  36.  
  37.     public static final Creator<User> CREATOR = new Creator<User>() { 
  38.         @Override 
  39.         public User createFromParcel(Parcel in) { 
  40.             return new User(in); 
  41.         } 
  42.  
  43.         @Override 
  44.         public User[] newArray(int size) { 
  45.             return new User[size]; 
  46.         } 
  47.     }; 
  48.  
  49.  
  50.     // 几乎在所有的情况下都应该返回0,只有在当前对象中存在文件描述的时候,此方法返回CONTENTS_FILE_DESCRIPTOR(常量值为1) 
  51.     @Override 
  52.     public int describeContents() { 
  53.         return 0; 
  54.     } 
  55.  
  56.     // 将对象写入序列化结构中,其中i标识有两种值,0或者1(PARCELABLE_WRITE_RETURN_VALUE) 
  57.     // 为1时表示当前对象需要作为返回值返回,不能立即释放资源,几乎所有情况都为0 
  58.     @Override 
  59.     public void writeToParcel(Parcel parcel, int i) { 
  60.         parcel.writeInt(userId); 
  61.         parcel.writeString(userName); 
  62.         // 注意这里,并不是直接写入boolean值,而是写入整数值 
  63.         parcel.writeByte((byte) (isMale ? 1 : 0)); 
  64.         // 此为不同之处2 
  65.         parcel.writeParcelable(book, i); 
  66.     } 
  67.  
  68.     @Override 
  69.     public String toString() { 
  70.         return "User{ " + 
  71.                 "userId = " + userId + 
  72.                 ", userName = " + userName + 
  73.                 ", isMale = " + isMale + 
  74.                 "book = " + book.toString() + 
  75.                 " }"
  76.     } 
  77.  
  78.  

可以看出,结果已经正确的打印了出来了:

注意:在 Parcelable 中,我们无法直接写入 boolean 值,而是将其转化为整数值进行保存,这里为 Byte,当然,你也可以使用 Int 等。



作者:i_seek_u
来源:51CTO
目录
相关文章
|
6天前
|
存储 Java
Java编程中的对象序列化与反序列化
【9月更文挑战第12天】在Java的世界里,对象序列化与反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何通过实现Serializable接口来标记一个类的对象可以被序列化,并探索ObjectOutputStream和ObjectInputStream类的使用,以实现对象的写入和读取。我们还将讨论序列化过程中可能遇到的问题及其解决方案,确保你能够高效、安全地处理对象序列化。
|
22天前
|
存储 Java
Java编程中的对象序列化与反序列化
【8月更文挑战第28天】在Java世界中,对象序列化与反序列化是数据持久化和网络传输的关键技术。本文将深入浅出地探讨这一过程,带你领略其背后的原理及应用,让你的程序在数据的海洋中自由航行。
|
16天前
|
存储 Java
Java编程中的对象序列化与反序列化
【9月更文挑战第2天】在Java的世界里,对象序列化和反序列化就像是给数据穿上了一件隐形的斗篷。它们让数据能够轻松地穿梭于不同的系统之间,无论是跨越网络还是存储在磁盘上。本文将揭开这层神秘的面纱,带你领略序列化和反序列化的魔法,并展示如何通过代码示例来施展这一魔法。
14 0
|
1月前
|
存储 安全 Java
揭秘Java序列化神器Serializable:一键解锁对象穿越时空的超能力,你的数据旅行不再受限,震撼登场!
【8月更文挑战第4天】Serializable是Java中的魔术钥匙,开启对象穿越时空的能力。作为序列化的核心,它让复杂对象的复制与传输变得简单。通过实现此接口,对象能被序列化成字节流,实现本地存储或网络传输,再通过反序列化恢复原状。尽管使用方便,但序列化过程耗时且存在安全风险,需谨慎使用。
35 7
|
2月前
|
存储 Java
JaveSE—IO流详解:对象输入输出流(序列化及反序列化)
JaveSE—IO流详解:对象输入输出流(序列化及反序列化)
|
2月前
|
存储 Java 开发者
Java中的对象序列化详解
Java中的对象序列化详解
|
2月前
|
存储 JSON 数据库
项目管理定义问题之什么是序列化大对象的值对象数据库形态
项目管理定义问题之什么是序列化大对象的值对象数据库形态
|
3月前
|
XML 机器学习/深度学习 存储
技术心得:对象的序列化存入数据库,与反序列化
技术心得:对象的序列化存入数据库,与反序列化
33 0
|
4月前
对象数组序列化和反序列化
对象数组序列化和反序列化
32 1
|
4月前
|
存储 JSON 安全
Python中对象到文件的序列化和反序列化
【4月更文挑战第2天】在Python编程中,序列化和反序列化是处理对象与文件之间转换的重要技术。序列化是将对象状态转换为可以存储或传输的形式的过程,通常是将对象转换为字节流。反序列化则是将序列化后的形式转换回对象的过程。在Python中,我们可以使用`pickle`模块来轻松地实现对象的序列化和反序列化。