一站式的消息管理器

123 阅读7分钟

一站式的消息管理器

在网络应用中,消息处理是必不可少的,该文章主要简单介绍一款简单的消息管理器的实现,其具备以下功能:

  • 提供多种消息序列化和反序列化方式,目前支持JDK、ProtoStuff以及JSON,提供其他自定义的序列化/反序列化器插口。
  • 提供多种消息加密/解密,目前支持对称加密:AES、不对称加密:RSA、不可逆散列加密:MD5以及Base64的转化并且提供自定义的加密/解密器插口。
  • 提供简单方便的消息构造,支持链式构造。
  • 支持创建可复用的流程化消息发送器和接收器、可以完成从消息发布、消息格式化、消息序列化/反序列化,消息加密/解密的流程处理,并且发送器和接收器支持序列化保存。

首先需要引入依赖:

 <dependency>
       <groupId>io.protostuff</groupId>
       <artifactId>protostuff-core</artifactId>
       <version>1.6.0</version>
 </dependency>
 <dependency>
       <groupId>io.protostuff</groupId>
       <artifactId>protostuff-runtime</artifactId>
       <version>1.6.0</version>
 </dependency>
 <dependency>
       <groupId>com.google.code.gson</groupId>
       <artifactId>gson</artifactId>
       <version>2.8.5</version>
 </dependency>

然后直接拷贝源码

 import com.google.gson.*;
 import io.protostuff.LinkedBuffer;
 import io.protostuff.ProtostuffIOUtil;
 import io.protostuff.Schema;
 import io.protostuff.runtime.RuntimeSchema;
 import sun.misc.BASE64Decoder;
 import sun.misc.BASE64Encoder;
 ​
 import javax.crypto.Cipher;
 import javax.crypto.spec.SecretKeySpec;
 import java.io.*;
 import java.lang.reflect.Type;
 import java.math.BigInteger;
 import java.nio.charset.Charset;
 import java.nio.charset.StandardCharsets;
 import java.security.*;
 import java.security.spec.PKCS8EncodedKeySpec;
 import java.security.spec.X509EncodedKeySpec;
 import java.util.*;
 import java.util.concurrent.ConcurrentHashMap;
 ​
 /**
  * 提供一站式的消息处理
  *
  * @author JingGe(* ^ ▽ ^ *)
  * @date 2023-03-24 15:53
  * @email 1158055613@qq.com
  */
 public class MessageManager {
     /*=================消息序列化/反序列化器=================>*/
     private static final Gson GSON = new GsonBuilder()
             .enableComplexMapKeySerialization()
             .setDateFormat("yyyy-MM-dd HH:mm:ss")
             .registerTypeAdapter(Class.class, new ClassCodec())
             .create();
     public static final MessageSerializer<String> JSON_SERIALIZER = GSON::toJson;
     public static final MessageDeserializer<Object> JSON_DESERIALIZER = (source, type) -> GSON.fromJson(new String(source, StandardCharsets.UTF_8), type);
     public static final MessageSerializer<byte[]> PROTO_STUFF_SERIALIZER = ProtoStuffUtil::serialize;
     public static final MessageDeserializer<Object> PROTO_STUFF_DESERIALIZER = ProtoStuffUtil::deserialize;
     public static final MessageSerializer<byte[]> JDK_SERIALIZER = source -> {
         ByteArrayOutputStream stream = new ByteArrayOutputStream();
         try {
             ObjectOutputStream outputStream = new ObjectOutputStream(stream);
             outputStream.writeObject(source);
         } catch (IOException e) {
             e.printStackTrace();
         }
 ​
         return stream.toByteArray();
     };
     public static final MessageDeserializer<Object> JDK_DESERIALIZER = (source, type) -> {
         ByteArrayInputStream stream = new ByteArrayInputStream(source);
         try {
             ObjectInputStream inputStream = new ObjectInputStream(stream);
             return (Serializable) inputStream.readObject();
         } catch (IOException | ClassNotFoundException e) {
             e.printStackTrace();
         }
         return null;
     };
 ​
     public interface MessageSerializer<T> {
         T serialize(Object source);
     }
 ​
     public interface MessageDeserializer<T> {
         T deserialize(byte[] source, Class<T> type);
     }
     /*=======================Finished======================<*/
 ​
     /*=================加密/解密器=================>*/
     public static final Encryptor<Object> NO_ENCRYPTOR = (source, key) -> source;
     public static final Decipher<Object> NO_DECIPHER = (source, key) -> source;
     public static final Encryptor<byte[]> AES_ENCRYPTOR = (source, key) -> {
         try {
             Cipher cipher = Cipher.getInstance("AES/ECB/PKCS5Padding");
             cipher.init(Cipher.ENCRYPT_MODE, new SecretKeySpec(bytes(key, Charset.defaultCharset(), "密钥只能是byte array或者String"), "AES"));
             return cipher.doFinal(source);
         } catch (Exception e) {
             e.printStackTrace();
         }
         return null;
     };
     public static final Decipher<byte[]> AES_DECIPHER = (source, key) -> {
         try {
             Cipher cipher = Cipher.getInstance("AES/ECB/PKCS5Padding");
             cipher.init(Cipher.DECRYPT_MODE, new SecretKeySpec(bytes(key, Charset.defaultCharset(), "密钥只能是byte array或者String"), "AES"));
             return cipher.doFinal(source);
         } catch (Exception e) {
             e.printStackTrace();
         }
         return null;
     };
     public static final RSAEncryptor RSA_ENCRYPTOR = new RSAEncryptor();
     public static final RSADecipher RSA_DECIPHER = new RSADecipher();
 ​
     public static final MD5Encryptor MD5_ENCRYPTOR = new MD5Encryptor();
 ​
     public static final Encryptor<String> BASE64_ENCRYPTOR = (source, key) -> {
         BASE64Encoder encoder = new BASE64Encoder();
         return encoder.encode(source);
     };
     public static final Decipher<byte[]> BASE64_DECIPHER = (source, key) -> {
         try {
             BASE64Decoder decoder = new BASE64Decoder();
             return decoder.decodeBuffer(new String(source));
         } catch (Exception e) {
             e.printStackTrace();
         }
         return null;
     };
 ​
     public static class RSAEncryptor implements Encryptor<byte[]> {
 ​
         @Override
         public byte[] encrypt(byte[] source, Object publicKey) {
             try {
                 PublicKey key = null;
                 if (publicKey instanceof PublicKey) {
                     key = (PublicKey) publicKey;
                 } else {
                     byte[] content = bytes(publicKey, Charset.defaultCharset(), "密钥格式异常:" + publicKey
                             .getClass()
                             .getName());
                     X509EncodedKeySpec keySpec = new X509EncodedKeySpec(content);
                     KeyFactory keyFactory = KeyFactory.getInstance("RSA");
                     key = keyFactory.generatePublic(keySpec);
                     ;
                 }
                 Cipher cipher = Cipher.getInstance("RSA");
                 cipher.init(Cipher.ENCRYPT_MODE, key);
                 return cipher.doFinal(source);
             } catch (Exception e) {
                 e.printStackTrace();
             }
             return null;
         }
 ​
         public KeyPair generateKeyPair(int keySize) {
             try {
                 keySize = keySize <= 0 ? 2048 : keySize;
                 KeyPairGenerator keyPairGenerator = KeyPairGenerator.getInstance("RSA");
                 keyPairGenerator.initialize(keySize);
                 return keyPairGenerator.generateKeyPair();
             } catch (Exception e) {
                 e.printStackTrace();
             }
             return null;
         }
 ​
 ​
     }
 ​
     public static class RSADecipher implements Decipher<byte[]> {
         @Override
         public byte[] decrypt(byte[] source, Object privateKey) {
             try {
                 PrivateKey key = null;
                 if (privateKey instanceof PrivateKey) {
                     key = (PrivateKey) privateKey;
                 } else {
                     byte[] content = bytes(privateKey, Charset.defaultCharset(), "密钥格式异常:" + privateKey
                             .getClass()
                             .getName());
                     PKCS8EncodedKeySpec keySpec = new PKCS8EncodedKeySpec(content);
                     KeyFactory keyFactory = KeyFactory.getInstance("RSA");
                     key = keyFactory.generatePrivate(keySpec);
                 }
                 Cipher cipher = Cipher.getInstance("RSA");
                 cipher.init(Cipher.DECRYPT_MODE, key);
                 return cipher.doFinal(source);
             } catch (Exception e) {
                 e.printStackTrace();
             }
             return null;
         }
     }
 ​
     public static class MD5Encryptor implements Encryptor<String> {
         @Override
         public String encrypt(byte[] source, Object saltKey) {
             byte[] key = bytes(saltKey, Charset.defaultCharset(), "密钥只能是String或byte array");
             byte[] content = new byte[source.length + key.length];
             System.arraycopy(source, 0, content, 0, source.length);
             System.arraycopy(key, 0, content, source.length, key.length);
             byte[] res;
             try {
                 MessageDigest messageDigest = MessageDigest.getInstance("md5");
                 res = messageDigest.digest(messageDigest.digest(content));
                 if (res != null) {
                     StringBuilder md5code = new StringBuilder(new BigInteger(1, res).toString(16));
                     for (int i = 0; i < 32 - md5code.length(); i++) {
                         md5code.insert(0, "0");
                     }
                     return md5code.toString();
                 }
             } catch (NoSuchAlgorithmException e) {
                 e.printStackTrace();
             }
             return null;
         }
     }
 ​
     public interface Encryptor<T> {
         T encrypt(byte[] source, Object key);
     }
 ​
     public interface Decipher<T> {
         T decrypt(byte[] source, Object key);
     }
     /*=======================Finished======================<*/
 ​
     /*=================消息格式化器=================>*/
 ​
 ​
     public interface MessageFormatter<T> {
         void format(T msg);
     }
     /*=======================Finished======================<*/
 ​
     public static MessageBuilder newMessageBuilder() {
         return new MessageBuilder();
     }
 ​
     public static <T> MessageSenderWrapper<T> newMessageSenderWrapper(MessageSender<T> sender) {
         return new MessageSenderWrapper<>(sender);
     }
 ​
     public static <T> MessageReceiverWrapper<T> newMessageReceiverBuilder(MessageReceiver<T> receiver) {
         return new MessageReceiverWrapper<>(receiver);
     }
 ​
     public static <R> R getMessage(Object msg, MessageSerializer<R> serializer) {
         return serializer.serialize(msg);
     }
 ​
     public static <R> R encryptMessage(byte[] msg, byte[] key, Encryptor<R> encryptor) {
         return encryptor.encrypt(msg, key);
     }
 ​
     public static <R> R decryptMessage(byte[] msg, byte[] key, Decipher<R> decipher) {
         return decipher.decrypt(msg, key);
     }
 ​
     public static String convert2Base64(byte[] msg) {
         return BASE64_ENCRYPTOR.encrypt(msg, null);
     }
 ​
     public static byte[] base642Bytes(String base64) {
         return BASE64_DECIPHER.decrypt(base64.getBytes(), null);
     }
 ​
     @SuppressWarnings("unchecked")
     public static <T> T fromMessage(byte[] msg, Class<T> type, MessageDeserializer<Object> deserializer) {
         return (T) deserializer.deserialize(msg, (Class<Object>) type);
     }
 ​
     public static Map<String, Object> wrapper2Map(Code code, String message, Object data) {
         Map<String, Object> msg = new HashMap<>();
         msg.put("code", code.flag);
         msg.put("message", message);
         msg.put("data", data);
         return msg;
     }
 ​
     public static String getJsonMessage(Code code, String message, Object data) {
         return getMessage(wrapper2Map(code, message, data), JSON_SERIALIZER);
     }
 ​
     public static String getJsonMessage(Code code, String message) {
         return getMessage(wrapper2Map(code, message, null), JSON_SERIALIZER);
     }
 ​
     public static byte[] getProtoStuffMessage(Code code, String message, Object data) {
         return getMessage(wrapper2Map(code, message, data), PROTO_STUFF_SERIALIZER);
     }
 ​
     public static byte[] getProtoStuffMessage(Code code, String message) {
         return getMessage(wrapper2Map(code, message, null), PROTO_STUFF_SERIALIZER);
     }
 ​
     public static byte[] getJDKMessage(Code code, String message, Object data) {
         return getMessage(wrapper2Map(code, message, data), JDK_SERIALIZER);
     }
 ​
     public static byte[] getJDKMessage(Code code, String message) {
         return getMessage(wrapper2Map(code, message, null), JDK_SERIALIZER);
     }
 ​
     public static class MessageBuilder {
         private final Map<String, Object> container = new ConcurrentHashMap<>();
         private MessageSerializer<?> serializer;
 ​
         private MessageBuilder() {
         }
 ​
         public MessageBuilder addMsg(String title, Object content) {
             container.put(title, content);
             return this;
         }
 ​
         public MessageBuilder addAllMsg(Map<String, Object> msg) {
             container.putAll(msg);
             return this;
         }
 ​
         public MessageBuilder code(Code code) {
             container.put("code", code.flag);
             return this;
         }
 ​
         public MessageBuilder message(String message) {
             container.put("message", message);
             return this;
         }
 ​
         public MessageBuilder data(Object data) {
             container.put("data", data);
             return this;
         }
 ​
         public MessageBuilder setSerializer(MessageSerializer<?> serializer) {
             this.serializer = serializer;
             return this;
         }
 ​
         public void clear() {
             container.clear();
         }
 ​
         @SuppressWarnings("unchecked")
         public <T> T getFinalMessage() {
             return (T) (serializer == null ? JSON_SERIALIZER : serializer).serialize(container);
         }
     }
 ​
     public static class MessageSender<T> implements Serializable, MessageHandler<T> {
         private final List<MessageFormatter<Object>> formatters = new ArrayList<>();
         private MessageSerializer<?> serializer;
         private Encryptor<?> encryptor;
         private Object key;
         private Charset charset;
         private boolean encode2Base64WhenEncrypted;
 ​
         public MessageSender() {
         }
 ​
         @Override
         @SuppressWarnings("unchecked")
         public T getFinalMessage(Object msg) {
             if (msg == null) {
                 return null;
             }
             for (MessageFormatter<Object> formatter : formatters) {
                 if (formatter != null) {
                     formatter.format(msg);
                 }
             }
             Object ser = serializer == null ? msg : serializer.serialize(msg);
             byte[] content = bytes(ser, charset, "序列化后的消息只能是String或者byte array");
             if (encryptor != null) {
                 if (encode2Base64WhenEncrypted) {
                     return (T) convert2Base64(bytes(encryptor.encrypt(content, key), charset, "加密后的消息只能是String或者byte array"));
                 }
                 return (T) encryptor.encrypt(content, key);
             }
             return (T) ser;
         }
 ​
         @SuppressWarnings("unchecked")
         public static <T> MessageSender<T> fromBytes(byte[] source, MessageDeserializer<Object> deserializer) {
             return (MessageSender<T>) fromMessage(source, MessageSender.class, deserializer);
         }
     }
 ​
     public static class MessageReceiver<T> implements Serializable, MessageHandler<T> {
         private final List<MessageFormatter<Object>> formatters = new ArrayList<>();
         private MessageDeserializer<Object> deserializer;
         private Decipher<?> decipher;
         private Class<T> type;
         private Object key;
         private Charset charset;
         private boolean decode2Base64BeforeDecrypt;
 ​
         public MessageReceiver() {
         }
 ​
         @Override
         @SuppressWarnings("unchecked")
         public T getFinalMessage(Object msg) {
             byte[] content = null;
             if (decode2Base64BeforeDecrypt) {
                 if (msg instanceof String) {
                     content = base642Bytes((String) msg);
                 } else {
                     throw new UnsupportedOperationException("消息不是String类型,无法进行Base64解码");
                 }
             }
             Object des = content == null ? msg : content;
             if (decipher != null) {
                 des = decipher.decrypt(content == null ? bytes(msg, charset, "如果消息需要解密,则消息必须是byte array") : content, key);
             }
             T ser = (T) des;
             if (deserializer != null) {
                 ser = (T) deserializer.deserialize(bytes(des, charset, "解密后的结果必须是String或者byte array"), (Class<Object>) type);
             }
             for (MessageFormatter<Object> formatter : formatters) {
                 if (formatter != null) {
                     formatter.format(ser);
                 }
             }
             return (T) ser;
         }
 ​
         @SuppressWarnings("unchecked")
         public static <T> MessageReceiver<T> fromBytes(byte[] source, MessageDeserializer<Object> deserializer) {
             return (MessageReceiver<T>) fromMessage(source, MessageReceiver.class, deserializer);
         }
     }
 ​
     public interface MessageHandler<T> {
         T getFinalMessage(Object msg);
     }
 ​
     private static byte[] bytes(Object src, Charset charset, String errorMsg) {
         if (src instanceof byte[]) {
             return (byte[]) src;
         } else if (src instanceof String) {
             return charset == null ? ((String) src).getBytes() : ((String) src).getBytes(charset);
         }
         throw new UnsupportedOperationException(errorMsg);
     }
 ​
     public static class MessageSenderWrapper<T> {
         private final MessageSender<T> sender;
 ​
         private MessageSenderWrapper(MessageSender<T> sender) {
             this.sender = sender;
         }
 ​
         public MessageSenderWrapper<T> addFormatter(MessageFormatter<Object> formatter) {
             sender.formatters.add(formatter);
             return this;
         }
 ​
         public MessageSenderWrapper<T> setSerializer(MessageSerializer<?> serializer) {
             sender.serializer = serializer;
             return this;
         }
 ​
         public MessageSenderWrapper<T> setEncryptor(Encryptor<?> encryptor, Object key, Charset charset, boolean encode2Base64) {
             if (encryptor == null || key == null) {
                 throw new IllegalArgumentException("加密器和密钥不能为空");
             }
             sender.encryptor = encryptor;
             sender.key = key;
             sender.charset = charset;
             sender.encode2Base64WhenEncrypted = encode2Base64;
             return this;
         }
 ​
         public MessageSender<T> wrap() {
             return sender;
         }
     }
 ​
     public static class MessageReceiverWrapper<T> {
         private final MessageReceiver<T> receiver;
 ​
         public MessageReceiverWrapper(MessageReceiver<T> receiver) {
             this.receiver = receiver;
         }
 ​
         public MessageReceiverWrapper<T> addFormatter(MessageFormatter<Object> formatter) {
             receiver.formatters.add(formatter);
             return this;
         }
 ​
         public MessageReceiverWrapper<T> setDeserializer(MessageDeserializer<Object> deserializer, Class<T> type) {
             receiver.deserializer = deserializer;
             receiver.type = type;
             return this;
         }
 ​
         public MessageReceiverWrapper<T> setDecipher(Decipher<?> decipher, Object key, Charset charset, boolean decodeBase642ByteArray) {
             if (decipher == null || key == null) {
                 throw new IllegalArgumentException("解密器和密钥不能为空");
             }
             receiver.decipher = decipher;
             receiver.key = key;
             receiver.charset = charset;
             receiver.decode2Base64BeforeDecrypt = decodeBase642ByteArray;
             return this;
         }
 ​
         public MessageReceiver<T> wrap() {
             return receiver;
         }
     }
 ​
     /**
      * 状态枚举,支持枚举标识修改
      *
      * @author Huang Yongxiang
      * @date 2021/9/30 9:25
      */
     public enum Code {
         /**
          * 请求成功
          */
         OK("success", 200),
         /**
          * 服务器内部错误
          */
         ERROR("error", 500),
         /**
          * 客户端请求的语法错误,服务器无法理解
          */
         BAD_REQUEST("bad request", 400),
         /**
          * 服务器理解请求客户端的请求,但是拒绝执行此请求
          */
         FORBIDDEN("Forbidden", 403),
         /**
          * 超时
          */
         TIME_OUT("time out", 408),
         /**
          * 服务器无法根据客户端请求的内容特性完成请求
          */
         NOT_ACCEPTABLE("Not Acceptable", 406),
         /**
          * 服务器不支持请求的功能,无法完成请求
          */
         NOT_SUPPORT("Not Implemented", 501),
         /**
          * 已接受。已经接受请求,但未处理完成
          */
         ACCEPTED("Accepted", 202);
         private String message;
         private Integer flag;
 ​
         Code(String message, Integer flag) {
             this.message = message;
             this.flag = flag;
         }
 ​
         public String getMessage() {
             return message;
         }
 ​
         public Code setMessage(String message) {
             this.message = message;
             return this;
         }
 ​
         public Integer getFlag() {
             return flag;
         }
 ​
         public Code setFlag(Integer flag) {
             this.flag = flag;
             return this;
         }
 ​
         @Override
         public String toString() {
             return "Code{" + "message='" + message + ''' + ", flag=" + flag + '}';
         }
     }
 ​
     private static class ClassCodec implements JsonSerializer<Class<?>>, JsonDeserializer<Class<?>> {
         @Override
         public Class<?> deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
             try {
                 String str = json.getAsString();
                 return Class.forName(str);
             } catch (ClassNotFoundException e) {
                 throw new JsonParseException(e);
             }
         }
 ​
         @Override
         public JsonElement serialize(Class<?> src, Type typeOfSrc, JsonSerializationContext context) {
             return new JsonPrimitive(src.getName());
         }
     }
 ​
     private static class ProtoStuffUtil {
         private static final ThreadLocal<LinkedBuffer> threadSafeBuffer = new ThreadLocal<>();
         private static final Map<Class<?>, Schema<?>> schemaMap = new ConcurrentHashMap<>();
 ​
         @SuppressWarnings("unchecked")
         public static <T> byte[] serialize(T source) {
             if (source == null) {
                 return null;
             }
             Class<T> clazz = (Class<T>) source.getClass();
             Schema<T> schema = getSchema(clazz);
             if (schema == null) {
                 throw new NullPointerException("无法获取" + clazz + "的模式");
             }
             try {
                 if (threadSafeBuffer.get() == null) {
                     threadSafeBuffer.set(LinkedBuffer.allocate(LinkedBuffer.DEFAULT_BUFFER_SIZE));
                 }
                 return ProtostuffIOUtil.toByteArray(source, schema, threadSafeBuffer.get());
             } finally {
                 threadSafeBuffer
                         .get()
                         .clear();
             }
         }
 ​
         public static <T> T deserialize(byte[] data, Class<T> type) {
             if (data == null || data.length == 0) {
                 return null;
             }
             Schema<T> schema = getSchema(type);
             if (schema == null) {
                 throw new NullPointerException("无法获取" + type + "的模式");
             }
             T source = schema.newMessage();
             ProtostuffIOUtil.mergeFrom(data, source, schema);
             return source;
         }
 ​
         @SuppressWarnings("unchecked")
         private static <T> Schema<T> getSchema(Class<T> clazz) {
             Schema<T> schema = (Schema<T>) schemaMap.get(clazz);
             if (schema == null) {
                 schema = RuntimeSchema.getSchema(clazz);
                 schemaMap.put(clazz, schema);
             }
             return schema;
         }
     }
 ​

可以增加一个main方法测试

 public static void main(String[] args) {
         //直接获取消息
         System.out.println(getJsonMessage(Code.OK, "请求成功"));
 ​
         //构建更加复杂的消息
         System.out.println((String) newMessageBuilder()
                 //修改默认Code的标识
                 .code(Code.OK.setFlag(0))
                 .message("成功")
                 .addMsg("totalNum", 1000)
                 .getFinalMessage());
 ​
         //对消息加密
         String message = "你好世界";
         String en = BASE64_ENCRYPTOR.encrypt(AES_ENCRYPTOR.encrypt(message.getBytes(), "autoJob!@#=123.?"), null);
         System.out.println("加密后:" + en);
         System.out.println("解密后:" + new String(AES_DECIPHER.decrypt(BASE64_DECIPHER.decrypt(en.getBytes(), null), "autoJob!@#=123.?")));
 ​
 ​
         System.out.println("/*=================创建可以复用的消息发送器和接收器=================>*/");
         //创建使用RSA加/解密的发送/接收器
         KeyPair keyPair = RSA_ENCRYPTOR.generateKeyPair(2048);
         MessageSender<String> sender = newMessageSenderWrapper(new MessageSender<String>())
                 //可以尝试使用其他序列化器
                 .setSerializer(PROTO_STUFF_SERIALIZER)
                 //可以尝试使用其他加密器
                 .setEncryptor(RSA_ENCRYPTOR, keyPair.getPublic(), Charset.defaultCharset(), true)
                 .wrap();
 ​
 ​
         MessageReceiver<String> receiver = newMessageReceiverBuilder(new MessageReceiver<String>())
                 //必须和发送器序列化器对应
                 .setDeserializer(PROTO_STUFF_DESERIALIZER, String.class)
                 //必须和发送器加密器对应
                 .setDecipher(RSA_DECIPHER, keyPair.getPrivate(), Charset.defaultCharset(), true)
                 .wrap();
         String msg = sender.getFinalMessage("你好世界");
         System.out.println(msg);
         System.out.println(receiver.getFinalMessage(msg));
         System.out.println("/*=======================Finished======================<*/");
 ​
         System.out.println("/*=================对发送器/接收器进行序列化保存=================>*/");
         //序列化
         byte[] senderContent = getMessage(sender, PROTO_STUFF_SERIALIZER);
         byte[] receiverContent = getMessage(receiver, PROTO_STUFF_SERIALIZER);
 ​
         //反序列化
         MessageSender<String> savedSender = MessageSender.fromBytes(senderContent, PROTO_STUFF_DESERIALIZER);
 ​
         MessageReceiver<String> savedReceiver = MessageReceiver.fromBytes(receiverContent, PROTO_STUFF_DESERIALIZER);
 ​
         //反序列化后发送消息
         String sMsg = savedSender.getFinalMessage("你好世界");
         System.out.println(sMsg);
         System.out.println(savedReceiver.getFinalMessage(sMsg));
         System.out.println("/*=======================Finished======================<*/");
     }
 }
 {"code":200,"message":"请求成功"}
 {"code":0,"totalNum":1000,"message":"成功"}
 加密后:PXDEasQsDfG4Ca0WLdO4Sg==
 解密后:你好世界
 /*=================创建可以复用的消息发送器和接收器=================>*/
 YeKSWWdYPu4BK9gAehP5cYLhBcRPaGFN4qlC8mtieRCSNjBuqj9uLb2B4ca/FmYGOxHTYgikUZ2w
 /ANc04Vrgz3isKF9VuA03hiW1oWpP3yIIROSji+ribhc6U0ihhelpqimrTQaaVi3mZ8AqrdPKGmM
 KSrc/DaHZq/xMzCm1jlucPixm5Nss8Qk9u0lZltKzx8X6vjhaWmLonF7RXL76fiIq6dA45C6JbEG
 33R87VMO0xYwIWenvPfwrELrJ/1fqoWFZIBCJWmrqIkulNW7REZqSkFd2pVwF+NH9abN5HH/QOl0
 Y3kwRuazZtydkchw0s80WyP1CZ9lC/RK5HKQpQ==
 你好世界
 /*=======================Finished======================<*/
 /*=================对发送器/接收器进行序列化保存=================>*/
 lDUZdBYoU86AKilWY1ZdQR9SS/ud1xxnKyUDaKgoMRFLEQqEd2lnD7LDROSfqRNSGwG8ubQ37Db+
 2FjiCYsZMoU9B41m5nhm96lu5sWJrhhVLN7zvjdW5G3i6xWuzQrI9S3RiXUCK2zYBh9r2CIm9iAM
 mQ9VMZJKYewjKDJYta+hKGcKVg7/lJnX+lu8EmXLuKeCKymXycXHY3VMeURJZQrgLsz9qrQYw/4u
 l1nREc2LsL9/TVP3un98XpjUJxTkeyYJ74iLti5FPP3WD+ObR3OTzEZ/9sZndTEKZYFOYlanqavR
 hotfSDDTah108C9gNsnnxYUnUasNqF3xOgMETw==
 你好世界
 /*=======================Finished======================<*/
 ​
 Process finished with exit code 0

如果对你有帮助,谢谢点赞、收藏、Star