Hadoop序列化机制及实例

风格不统一 提交于 2020-04-03 04:26:55

序列化


1、什么是序列化?
将结构化对象转换成字节流以便于进行网络传输或写入持久存储的过程。
2、什么是反序列化?
将字节流转换为一系列结构化对象的过程。
序列化用途
1、作为一种持久化格式。 
2、作为一种
通信的数据格式。 

3、作为一种数据拷贝、克隆机制。


Java序列化和反序列化

1、创建一个对象实现了Serializable 
2、序列化    :ObjectOutputStream.writeObject(序列化对象) 
      反序列化:ObjectInputStream .readObject()返回序列化对象 
具体实现,可参考如下文章: 

http://blog.csdn.net/scgaliguodong123_/article/details/45938555

为什么Hadoop不直接使用java序列化?

Hadoop的序列化机制与Java的序列化机制不同,它将对象序列化到流中,
值得一提的是java的序列化机制是不断的创建对象,

但在hadoop的序列化机制中,用户可以复用对象,这样就减少了java对象的分配和回收,提高了应用效率。

Hadoop序列化

Hadoop的序列化不采用java的序列化,而是实现了自己的序列化机制。 

Hadoop通过Writable接口实现的序列化机制,不过没有提供比较功能,所以和java中的Comparable接口合并,提供一个接口WritableComparable。(自定义比较)

Writable接口提供两个方法(write和readFields)。

需要进行比较的话,要实现WritableComparable接口。

比如mapreduce中需要对key值进行相应的排序。可参考下面的例子: 

http://blog.csdn.net/scgaliguodong123_/article/details/46010947


Hadoop提供了几个重要的序列化接口与实现类:

外部集合的比较器

WritableComparator 实现了 RawComparator<T> 继承了 extends Comparator<T>

 

实现了WritableComparable接口的类(自定义比较)

仅实现了Writable接口的类



自定义Writable

1、实现WritableComparable接口 
2、实现相应的接口方法: 

A. write(dos)             //将对象转换为字节流并写入到输出流out中。 

B. readFileds()     //从输入流in中读取字节流并发序列化为对象。 

C. compareTo(o) //将this对象和对象o进行比较。 

可参考下面的例子,自定义NewK2类: 
http://blog.csdn.net/scgaliguodong123_/article/details/46010947

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
 public static void main(String[] args) throws IOException {        
        Student student = new Student("liguodong"22"男");
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(new File("g:/liguodong.txt")));
        DataOutputStream dos = new DataOutputStream(bos);        
        student.write(dos);        
        dos.flush();        
        dos.close();        
        bos.close();        
         
        Student student2 = new Student();
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream(new File("g:/liguodong.txt")));
        DataInputStream dis = new DataInputStream(bis);        
        student2.readFields(dis);
        System.out.println("name="+student2.getName()
                +",age="+student2.getAge()+",sex="+student2.getSex());// 打印:name=liguodong,age=22,sex=男
}
  
// 省略
//Hadoop序列化机制及实例 - 琴弦上、漫步 - CSDN博客
//http://blog.csdn.net/scgaliguodong123_/article/details/46385761#t3

Hadoop序列化优势:

1、紧凑:Hadoop中最稀缺的资源是宽带,所以紧凑的序列化机制可以充分的利用宽带。 
2、快速:通信时大量使用序列化机制,因此,需要减少序列化和反序列化的开销。 
3、可扩展:随着通信协议的升级而可升级。 
4、互操作:支持不同开发语言的通信。 
Hadoop1.x 序列化仅满足了紧凑和快速的特点。

Hadoop序列化的作用

序列化在分布式环境的两大作用:进程间通信,永久存储。 
Hadoop节点间通信。

序列化框架

Apache Avro

1、丰富数据结构类型 
2、快速可压缩二进制数据形式 
3、存储持久数据的文件容器 
4、远程过程调用RPC 
5、简单的动态语言结合功能,Avro和动态语言结合后,读写数据文件和使用RPC协议都不需要生成代码,而代码生成作为一种可选的优化,只值得在静态类型语言中实现。

Facebook Thrift 

1、是一种可伸缩的跨语言服务的发展软件框架 
2、它结合了功能强大的软件堆栈的代码生成引擎,以建设服务,工作效率和无缝地与C++,C#,.Java,PythonPHP和Ruby结合。 
3、允许定义一个简单的定义文件中的数据类型和服务接口,以作为输入文件,编译器生成代码用来方便地生成RPC客户端和服务器通信的无缝跨编程语言

Google Protocolbuffer  

PB是Google开源的一种轻量级的结构化数据存储格式,可以用于结构化数据的序 
列化与反序列化,很适合做数据存储RPC数据交换格式 

优点: 
与 XML相比,它更小、更快、也更简单。你可以定义自己的数据结构,然后使用代码生成器生成的代码来读写这个数据结构。你甚至可以在无需重新部署程序的情况下更新数据结构。只需使用 Protobuf 对数据结构进行一次描述,即可利用各种不同语言或从各种不同数据流中对你的结构化数据轻松读写。

它有一个非常棒的特性,即“向后”兼容性好,人们不必破坏已部署的、依靠”老”数据格式的程序就可以对数据结构进行升级。这样您的程序就可以不必担心因为消息结构的改变而造成的大规模的代码重构或者迁移的问题。因为添加新的消息中的 field 并不会引起已经发布的程序的任何改变

Protocolbuffer语义更清晰无需类似 XML 解析器的东西(因为 Protobuf 编译器会将 .proto 文件编译生成对应的数据访问类以对 Protobuf 数据进行序列化、反序列化操作)。使用 Protobuf 无需学习复杂的文档对象模型,Protobuf 的编程模式比较友好,简单易学,同时它拥有良好的文档和示例,对于喜欢简单事物的人们而言,Protobuf 比其他的技术更加有吸引力。

不足: 
Protbuf 与 XML 相比也有不足之处。它功能简单,无法用来表示复杂的概念

由于文本并不适合用来描述数据结构,所以 Protobuf 也不适合用来对基于文本的标记文档(如 HTML)建模。另外,由于 XML 具有某种程度上的自解释性,它可以被人直接读取编辑,在这一点上 Protobuf 不行,它以二进制的方式存储,除非你有 .proto 定义,否则你没法直接读出 Protobuf 的任何内容。

Hadoop2.X用到的Protocolbuffer 
hadoop-2.6.0-src/hadoop-hdfs-project/hadoop-hdfs/src/main/proto



参考

Hadoop序列化机制及实例 - 琴弦上、漫步 - CSDN博客

http://blog.csdn.net/scgaliguodong123_/article/details/46385761#t3


hadoop深入研究:(十三)——序列化框架 - 独自登高楼 望断天涯路 - CSDN博客

http://blog.csdn.net/lastsweetop/article/details/9376495




易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!