架构师基础技能:序列化与反序列化,Android的Parcelable与Serializable区别是什么

本文详细探讨了Android中的序列化与反序列化,对比了Serializable和Parcelable接口的使用。Parcelable提供了更高效的序列化性能,适合内存操作,而Serializable适合持久化存储。了解这两个接口在Android开发中的应用场景和选择依据是架构师必备的知识。
摘要由CSDN通过智能技术生成

本专栏专注分享大型Bat面试知识,后续会持续更新,喜欢的话麻烦点击一个关注

hermes架构的Github地址
可跨进程的EventBus 的Github地址
更多完整项目下载。未完待续。源码。图文知识后续上传github。
可以点击关于我 联系我获取
VX:mm14525201314

面试官: 序列化与反序列化的原理,Android的Parcelable与Serializable区别是什么
心理分析:序列化与反序列化在面试中算得上是一个重点的话题,有时候稍微不对,面试官会认为开发没多久
求职者:应该从Parcelable与Serializable的使用开始讲起

作为架构师必备的一些知识点,在这里分享我就列举一下我的准备的方向
架构师必备基础.jpg

一、序列化与反序列化

由于在系统底层,数据的传输形式是简单的字节序列形式传递,即在底层,系统不认识对象,只认识字节序列,而为了达到进程通讯的目的,需要先将数据序列化,而序列化就是将对象转化字节序列的过程。相反地,当字节序列被运到相应的进程的时候,进程为了识别这些数据,就要将其反序列化,即把字节序列转化为对象。有了以上理解,接下来我们认识两个用于序列化和反序列化的接口:Serializable接口和Parcelable接口。

二、Serializable接口

Java提供了一个序列化接口,serialable接口,该接口在文档中定义如下:Marks classes that can be serialized by ObjectOutputStream and deserialized by ObjectInputStream.从这句话可看出,该接口只是标记了当前类是可以序列化的,是一个空接口,仅仅提供了标志功能,具体的序列化与反序列化操作是由ObjectOutputStreamObjectInputStream完成的。   继续读文档,发现该接口要求我们在实现了该接口的类中声明如下的一个变量:

private static final long serialVersionUID= 1L; 

这个变量有什么用呢?试想一下,如果没有手动指定该值,一开始序列化了classA,得到文件A,接着对classA的内部结构更改,比如添加了一个新的变量,那么此时反序列化则会失败,因为实际上系统在序列化的时候,会自动计算出一个serialVersionUID值,并保存在已经序列化好的数据中,此时修改了classA,那么反序列化的时候系统就会重新计算一个新的serialVersionUID值,那么两个值就会不相等,就会反序列化失败。所以,手动指定一个值,能很大程度上保存数据,防止数据丢失。

接下来,我们来看一下序列化和反序列化的具体步骤:

  • 对象的序列化:   
    (1)实例化一个对象输出流:ObjectOutputStream,该对象输出流可以包装一个输出流,比如文件输出流。   
    (2)使用ObjectOutputStream.writeObject(obj)进行写对象。
  • 对象的反序列化:   
    (1)实例化一个对象输入流:ObjectInputStream,该对象输入流可以包装一个输入流,比如文件输入流。   
    (2)使用ObjectInputStream.readObject(obj)进行读对象。

以下是一个实现序列化与反序列化的范例:
①User类,被序列化的类:

package com.chenyu.serialable;  
  
import java.io.Serializable;  
  
public class User implements Serializable {  
  
    private static final long serialVersionUID = 1L;  
      
    public int id;  
    public String username;  
    public String email;  
      
    public User(int id, String username, String email) {  
        this.id = id;  
        this.username = username;  
        this.email = email;  
    }   
}

②Test测试类,测试序列化与反序列化是否成功:

package com.chenyu.serialable;  
  
import java.io.FileInputStream;  
import java.io.FileNotFoundException;  
import java.io.FileOutputStream;  
import java.io.IOException;  
import java.io.ObjectInputStream;  
import java.io.ObjectOutputStream;  
  
public class Test {  
    public static void main(String[] args) throws FileNotFoundException, IOException, ClassNotFoundException {  
        //实例化User类  
        User user =new User(1,"TestName","example@126.com");  
        //序列化过程  
        ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("test.txt"));  
        objectOutputStream.writeObject(user);  
        objectOutputStream.close();  
        System.out.println("序列化成功!");  
        //反序列化过程  
        ObjectInputStream objectInputStream =new ObjectInputStream(new FileInputStream("test.txt"));  
        User newUser = (User) objectInputStream.readObject();  
        objectInputStream.close();  
        System.out.println("反序列化成功!");  
        System.out.println("ID:"+newUser.id+"  username:"+newUser.username+"  Email:"+newUser.email);  
    }  
} 

运行Test.java,得到如下结果:

注意:

静态成员变量属于类,而不是对象,所以不会参与序列化;使用transient关键字标记的成员变量不参与序列化过程。

三、Parcelable接口

接下来我们要说的是Parcelable接口,该接口是Android提供的新的序列化方式。首先,先看官方文档对该接口的描述:

Interface for classes whose instances can be written to and restored from a Parcel. Classes implementing the Parcelable interface must also have a static field called CREATOR, which is an object implementing the Parcelable.Creator interface.

除了实现该接口的方法外,还需创建一个名叫CREATOR的静态对象,该对象实现了一个Parcelable.Creator的匿名内部类。以下是官方文档的一个类实现Parcelable接口的典型例子:

public class MyParcelable implements Parcelable {  
     private int mData;  
  
     public int describeContents() {  
         return 0;  
     }  
  
     public void writeToParcel(Parcel out, int flags) {  
         out.writeInt(mData);  
     }  
  
     public static final Parcelable.Creator<MyParcelable> CREATOR  
             = new Parcelable.Creator<MyParcelable>() {  
         public MyParcelable createFromParcel(Parcel in) {  
             return new MyParcelable(in);  
         }  
  
         public MyParcelable[] newArray(int size) {  
             return new MyParcelable[size];  
         }  
     };  
       
     private MyParcelable(Parcel in) {  
         mData = in.readInt();  
     }  
 } 

下面介绍一下以上各个方法的作用:

  • ①writeToParcel(Parcel out,int flags):将当前对象写入序列化结构之中。
  • ②createFromParcel(Parcel in):从序列化后的对象中创建原始对象
  • ③newArray(int size):创建指定长度的原始对象数组
  • ④MyParcelable(Parcel in):从序列化后的对象中创建原始对象

由以上各个方法可知,writeToParcel方法负责将对象序列化,而CREATOR负责数据的反序列化,只要你的类实现了Parcelable接口,并实现以上方法,那么就能自动地对对象进行序列化和反序列化了。

**注意:**在writeToParcel方法中,调用了out.writeInt(data)方法,如果当前类有多个属性,比如:int id,String name,String email,那么方法体可以写为:

out.writeInt(id);  
out.writeString(name);  
out.writeString(email); 

这样写后,在相应的MyParcelable(Parcel in)反序列化方法也必须如下写:

in.readInt();  
in.readString();  
in.readString(); 

即顺序应该一一对应,否则,取出来的数据将会出错。   到目前为止,介绍了Serialable接口和Parcelable接口,这是IPC机制中比较基础的概念,应熟练掌握。希望我的文章能对你们的学习起到帮助作用。

四 Parcelable和Serializable的区别:

android自定义对象可序列化有两个选择一个是Serializable和Parcelable

1.对象为什么需要序列化
  • 永久性保存对象,保存对象的字节序列到本地文件。
  • 通过序列化对象在网络中传递对象。
  • 通过序列化对象在进程间传递对象。
2.当对象需要被序列化时如何选择所使用的接口
  • 在使用内存的时候ParcelableSerializable的性能高。
  • Serializable在序列化的时候会产生大量的临时变量,从而引起频繁的GC(内存回收)。

3.Parcelable不能使用在将对象存储在磁盘上这种情况,因为在外界的变化下Parcelable不能很好的保证数据的持续性。
一个智能,新颖且易于使用的Android进程间通信(IPC)框架。(简单易用的安卓进展间通信IPC框架)

hermes架构的Github地址
可跨进程的EventBus 的Github地址
更多完整项目下载。未完待续。源码。图文知识后续上传github。
可以点击关于我 联系我获取

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
GeoPandas是一个开源的Python库,旨在简化地理空间数据的处理和分析。它结合了Pandas和Shapely的能力,为Python用户提供了一个强大而灵活的工具来处理地理空间数据。以下是关于GeoPandas的详细介绍: 一、GeoPandas的基本概念 1. 定义 GeoPandas是建立在Pandas和Shapely之上的一个Python库,用于处理和分析地理空间数据。 它扩展了Pandas的DataFrame和Series数据结构,允许在其中存储和操作地理空间几何图形。 2. 核心数据结构 GeoDataFrame:GeoPandas的核心数据结构,是Pandas DataFrame的扩展。它包含一个或多个列,其中至少一列是几何列(geometry column),用于存储地理空间几何图形(如点、线、多边形等)。 GeoSeries:GeoPandas中的另一个重要数据结构,类似于Pandas的Series,但用于存储几何图形序列。 二、GeoPandas的功能特性 1. 读取和写入多种地理空间数据格式 GeoPandas支持读取和写入多种常见的地理空间数据格式,包括Shapefile、GeoJSON、PostGIS、KML等。这使得用户可以轻松地从各种数据源中加载地理空间数据,并将处理后的数据保存为所需的格式。 2. 地理空间几何图形的创建、编辑和分析 GeoPandas允许用户创建、编辑和分析地理空间几何图形,包括点、线、多边形等。它提供了丰富的空间操作函数,如缓冲区分析、交集、并集、差集等,使得用户可以方便地进行地理空间数据分析。 3. 数据可视化 GeoPandas内置了数据可视化功能,可以绘制地理空间数据的地图。用户可以使用matplotlib等库来进一步定制地图的样式和布局。 4. 空间连接和空间索引 GeoPandas支持空间连接操作,可以将两个GeoDataFrame按照空间关系(如相交、包含等)进行连接。此外,它还支持空间索引,可以提高地理空间数据查询的效率。
SQLAlchemy 是一个 SQL 工具包和对象关系映射(ORM)库,用于 Python 编程语言。它提供了一个高级的 SQL 工具和对象关系映射工具,允许开发者以 Python 类和对象的形式操作数据库,而无需编写大量的 SQL 语句。SQLAlchemy 建立在 DBAPI 之上,支持多种数据库后端,如 SQLite, MySQL, PostgreSQL 等。 SQLAlchemy 的核心功能: 对象关系映射(ORM): SQLAlchemy 允许开发者使用 Python 类来表示数据库表,使用类的实例表示表中的行。 开发者可以定义类之间的关系(如一对多、多对多),SQLAlchemy 会自动处理这些关系在数据库中的映射。 通过 ORM,开发者可以像操作 Python 对象一样操作数据库,这大大简化了数据库操作的复杂性。 表达式语言: SQLAlchemy 提供了一个丰富的 SQL 表达式语言,允许开发者以 Python 表达式的方式编写复杂的 SQL 查询。 表达式语言提供了对 SQL 语句的灵活控制,同时保持了代码的可读性和可维护性。 数据库引擎和连接池: SQLAlchemy 支持多种数据库后端,并且为每种后端提供了对应的数据库引擎。 它还提供了连接池管理功能,以优化数据库连接的创建、使用和释放。 会话管理: SQLAlchemy 使用会话(Session)来管理对象的持久化状态。 会话提供了一个工作单元(unit of work)和身份映射(identity map)的概念,使得对象的状态管理和查询更加高效。 事件系统: SQLAlchemy 提供了一个事件系统,允许开发者在 ORM 的各个生命周期阶段插入自定义的钩子函数。 这使得开发者可以在对象加载、修改、删除等操作时执行额外的逻辑。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值