MessagePack 使用

MessagePack(https://msgpack.org/) 是一个基于二进制高效的对象序列化 Library 用于跨语言通信。它可以像 JSON 那样,在许多种语言之间交换结构对象;但是它比 JSON 更快速也更轻巧。 支持 Python、Ruby、Java、C/C++、Javascript 等众多语言。 比 Google Protocol Buffers 还要快 4 倍。

MessagePack 特点:编解码高效,性能高;序列化之后的码流小。

快速入门:https://github.com/msgpack/msgpack-java

一、环境准备

<dependency>
<groupId>org.msgpack</groupId>
<artifactId>msgpack</artifactId>
<version>0.6.12</version>
</dependency>

二、MessagePack 使用

/**
* 关于Msgpack官网上是这么说的,It's like JSON.but fast and small
* 1.单个对象使用 Msgpack <BR />
* 1.1 此对象必须要实现 Serializable 接口(java.io.Serializable) <BR />
* 1.2 此对象必须要使用 @Message 注解(org.msgpack.annotation.Message) <BR />
* 1.3 此对象属性中不能有用 transient 修饰的字段. <BR />
* 2.序列化List,Map(List接口)直接这么做msgpack不支持.
* @author: leigang
* @version: 2018-04-28
*/
public class MsgpackTest { private long time;
private MessagePack msgpack; private UserBean user = new UserBean("binarylei", 22, true, 2000.0); @Before
public void beforeTest() {
time = System.currentTimeMillis();
msgpack = new MessagePack();
//msgpack.register(UserBean.class);
} @After
public void afterTest() {
msgpack.unregister();
System.out.println(System.currentTimeMillis() - time);
} //1. MessagePack 入门
@Test
public void test() throws IOException {
List<String> src = Arrays.asList(new String[]{"java", "python", "c"}); // Serialize
byte[] raw = msgpack.write(src); // Deserialize directly using a template
List<String> dst1 = msgpack.read(raw, Templates.tList(Templates.TString));
System.out.println(dst1); // Or, Deserialze to Value then convert type.
Value dynamic = msgpack.read(raw);
List<String> dst2 = new Converter(dynamic).read(Templates.tList(Templates.TString));
System.out.println(dst2);
} //2. JavaBean 对象序列化
@Test
public void testObject() throws IOException {
byte[] bytes = msgpack.write(user);
UserBean newValue = msgpack.read(bytes, UserBean.class);
Assert.assertEquals(user.getName(), newValue.getName());
} //3. JavaBean 包含 List,Map 属性,使用方式同上
@Test
public void testBean() throws IOException {
ListUserBean src = new ListUserBean();
List<UserBean> lst = new ArrayList<>();
lst.add(user);
src.setList(lst); byte[] bytes = msgpack.write(src);
ListUserBean dst1 = msgpack.read(bytes, ListUserBean.class);
System.out.println(dst1.getList());
} //4. List
@Test
public void testList() throws IOException {
List<UserBean> lst = new ArrayList<UserBean>();
for (int i = 0; i < 10; i++) {
lst.add(user);
} byte[] bytes = msgpack.write(lst);
List<UserBean> newValue = msgpack.read(bytes, Templates.tList(msgpack.lookup(UserBean.class)));
Assert.assertEquals(lst.size(), newValue.size());
} //5. Set
@Test
public void testSet() throws IOException {
Set<UserBean> set = new HashSet<UserBean>();
for (int i = 0; i < 10; i++) {
set.add(user);
} byte[] bytes = msgpack.write(set);
Set<UserBean> newValue = msgpack.read(bytes, Templates.tSet(msgpack.lookup(UserBean.class)));
Assert.assertEquals(set.size(), newValue.size());
} //6. Map
@Test
public void testMap() throws IOException {
Map<String, UserBean> map = new HashMap<>();
for (int i = 0; i < 10; i++) {
map.put(String.valueOf(i), user);
} byte[] bytes = msgpack.write(map);
Map<String, UserBean> newValue = msgpack.read(bytes,
Templates.tMap(Templates.TString, msgpack.lookup(UserBean.class)));
Assert.assertEquals(map.size(), newValue.size());
}
}

测试用的 bean 对象如下:

@Message
public class UserBean implements Serializable { private String name;
private int age;
private boolean sex;
private double salary; // 省略 getter/setter/constructor ...
} @Message
public class ListUserBean {
public List<UserBean> list; public List<UserBean> getList() {
return list;
} public void setList(List<UserBean> list) {
this.list = list;
}
}

三、踩过的坑

(1) 问题:Object 无法序列化

问题详见:https://github.com/msgpack/msgpack-java/issues/4

org.msgpack.MessageTypeException: Cannot find template for class java.lang.Object class. Try to add @message annotation to the class or call MessagePack.register(Type).
at org.msgpack.template.TemplateRegistry.lookup(TemplateRegistry.java:220)
at org.msgpack.template.TemplateRegistry.lookupGenericTypeImpl(TemplateRegistry.java:258)
at org.msgpack.template.TemplateRegistry.lookupGenericType(TemplateRegistry.java:230)
at org.msgpack.template.TemplateRegistry.lookup(TemplateRegistry.java:180)

如下 Map<String, Object> 中的 Object 出错,解决方法就是用具体的类代替 Object

@Message
public class Header implements Serializable {
// ...
//private Map<String, Object> attachmet = new HashMap<>();
private Map<String, String> attachmet = new HashMap<>();
}

参考:

  1. https://msgpack.org/
  2. https://github.com/msgpack/msgpack-java

每天用心记录一点点。内容也许不重要,但习惯很重要!

MessagePack 使用的更多相关文章

  1. 新型序列化类库MessagePack,比JSON更快、更小的格式

    MessagePack is an efficient binary serialization format. It lets you exchange data among multiple la ...

  2. MessagePack, Protocol Buffers和Thrift序列化框架原理和比较说明

    MessagePack, Protocol Buffers和Thrift序列化框架原理和比较说明 http://www.open-open.com/lib/view/open1412731170858 ...

  3. messagePack编解码

    首先引入javassist-3.20.0-GA.jar与msgpack-0.6.12.jar两个包,然后就可以使用. package com.ming.netty.code; import java. ...

  4. Android 数据传输之MessagePack使用

    介绍过什么是MessagePack之后,就进行Android与MessagePack的使用. 在MessagePack的官网上介绍MessagePack与Java结合使用的都是使用Maven作为JAR ...

  5. 计算机程序的思维逻辑 (63) - 实用序列化: JSON/XML/MessagePack

    上节,我们介绍了Java中的标准序列化机制,我们提到,它有一些重要的限制,最重要的是不能跨语言,实践中经常使用一些替代方案,比如XML/JSON/MessagePack. Java SDK中对这些格式 ...

  6. 基于.NET CORE微服务框架 -surging 基于messagepack、protobuffer、json.net 性能对比

    1.前言 surging内部使用的是高性能RPC远程服务调用,如果用json.net序列化肯定性能上达不到最优,所以后面扩展了protobuf,messagepack序列化组件,以支持RPC二进制传输 ...

  7. 快速序列化组件MessagePack介绍

    简介 MessagePack for C#(MessagePack-CSharp)是用于C#的极速MessagePack序列化程序,比MsgPack-Cli快10倍,与其他所有C#序列化程序相比,具有 ...

  8. Messagepack原理

    什么是Messagepack? 用官方的话说:MessagePack是一种高效的二进制序列化格式.它允许您像JSON一样在多个语言之间交换数据.但是,它更快并且更小.小整数被编码为一个字节,和典型的短 ...

  9. MessagePack简析

    一.MessagePack是什么 先看官方的定义:MessagePack是一种高效的二进制序列化格式.它允许您像JSON一样在多个语言之间交换数据.但是,它更快并且更小. 从官方定义中,可以有如下的结 ...

  10. 基于.NET CORE微服务框架 -谈谈surging 的messagepack、protobuffer、json.net 序列化

    1.前言 surging内部使用的是高性能RPC远程服务调用,如果用json.net序列化肯定性能上达不到最优,所以后面扩展了protobuf,messagepack序列化组件,以支持RPC二进制传输 ...

随机推荐

  1. opencv小问题大智慧

    opencv易错点记录 1. 判断条件不应使用CV_ASSERT(),而使用CV_Assert(). 2.  用下面这个自适应阈值必须进行数据的转换,不能直接传入数据!不然程序一直崩溃,找了很久才发现 ...

  2. tomcat原理分析与简单实现

    tomcat原理分析与简单实现 https://blog.csdn.net/u014795347/article/details/52328221 2016年08月26日 14:48:18 卫卫羊习习 ...

  3. uva-639-枚举

    题意: 象棋里的車可以吃横竖的車,题目加了一个墙,用于阻断攻击,问4x4的棋盘最多可以放多少只車, 思路:枚举每一个点,2^16次方种情况 #include<stdio.h> #inclu ...

  4. 搜集几个API接口

    新浪:http://int.dpool.sina.com.cn/iplookup/iplookup.php?format=js 返回数据:var remote_ip_info = {"ret ...

  5. leetcode944

    public class Solution { public int MinDeletionSize(string[] A) { ; ; j < A[].Length; j++) { ; i & ...

  6. leetcode227

    class Solution { public: stack<int> OPD; stack<char> OPR; int calculate(int num1, int nu ...

  7. jquery checkbox

    $(document).ready(function() { var $cr = $("#cr"); var cr = $cr[0]; $cr.click(function() { ...

  8. 12 python json&pickle&shelve模块

      1.什么叫序列化 序列化是指把内存里的数据类型转变成字符串,以使其能存储到硬盘或通过网络传输到远程,因为硬盘或网络传输时只能接受bytes(字节) 2.用于序列化的两个模块,json和pickle ...

  9. UI5-文档-4.27-Mock Server Configuration

    我们只是在一个真实的服务上运行我们的应用程序,但是对于开发和测试我们的应用程序,我们不希望依赖于“真实”服务的可用性,或者在数据服务所在的系统上增加额外的负载. 这个系统就是所谓的后端系统,我们现在将 ...

  10. 关于str.split(",")中间 什么时候该加\\转义

    java 分割符,对于某些符号的分割符进行切割的时候需要加转义字符,我贴上例子 1. package test1; public class split { public static void ma ...