一站式消息管理器

来源:https://www.cnblogs.com/hyxl-top/archive/2023/03/28/17264296.html
-Advertisement-
Play Games

一站式的消息管理器 在網路應用中,消息處理是必不可少的,該文章主要簡單介紹一款簡單的消息管理器的實現,其具備以下功能: 提供多種消息序列化和反序列化方式,目前支持JDK、ProtoStuff以及JSON,提供其他自定義的序列化/反序列化器插口。 提供多種消息加密/解密,目前支持對稱加密:AES、不對 ...


一站式的消息管理器

在網路應用中,消息處理是必不可少的,該文章主要簡單介紹一款簡單的消息管理器的實現,其具備以下功能:

  • 提供多種消息序列化和反序列化方式,目前支持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 [email protected]
 */
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

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • TS:是JS的超集,即對JS的擴展,主要提供了類型系統和對ES6+的支持,但TS最終會轉換為js代碼去執行。 特點: 1. 始於JavaScript, 歸於JavaScript 2. 強大的類型系統 3. 先進的JavaScript | | TypeScript | JavaScript | | | ...
  • Vue.js 是一個前端開發框架,它可以幫助我們快速構建單頁應用和複雜的交互界面。而 Webpack 則是一個前端資源打包工具,它可以將多個 JavaScript、CSS、HTML、圖片等資源打包成一個或多個文件,並提供模塊化載入、壓縮、混淆等功能,以便於前端開發和部署。 在使用 Vue.js 開發 ...
  • 作為 C 端前端研發,除了攻剋業務難點以外,也要有更深層的自我目標,那就是性能優化。這事兒說大不大,說小也不小,但難度絕對不一般,所涉及的範圍優化點深入工程每個細胞。做好前端性能優化絕非簡單之事!文章主要內容介紹前端性能考核指標及優化方案。 ...
  • Vue Router 是 Vue.js 的官方路由管理器,它可以幫助我們實現單頁應用(Single Page Application,SPA)的前端路由。 Vue Router 基於 Vue.js 的組件化思想,將路由視為組件,並提供了多種方式來定義路由和渲染組件。Vue Router 可以通過 U ...
  • 解釋器模式(Interpreter Pattern):是一種行為型設計模式,它可以用來解決一些特定問題,例如編譯器、計算器等等。這種模式定義了一個語言的語法,並用一個解釋器來解釋語言中的表達式。 解釋器模式可以用來處理例如數據格式化、表單驗證等業務場景。在這些場景中,我們需要定義一些語法規則,然後使 ...
  • Axios是一個基於Promise的HTTP客戶端,用於在瀏覽器和Node.js中發送HTTP請求。它可以使用在Vue中發送請求以及與後端API進行交互。 在Vue中使用Axios可以通過以下步驟: 安裝Axios 可以通過npm或yarn來安裝Axios:npm install axios yar ...
  • 3D 全景技術可以實現日常生活中的很多功能需求,比如地圖的街景全景模式、數字展廳、線上看房、社交媒體的全景圖預覽、短視頻直播平臺的全景直播等。Three.js 實現全景功能也是十分方便的,本文將介紹如何一步步優雅實現在多個3D全景中穿梭漫游,達到如在真實世界中前進後退的視覺效果。全景漫游系列文章將分... ...
  • XSS攻擊是什麼? XSS攻擊是指攻擊者利用網站中的漏洞,向頁面中註入惡意腳本,從而獲取用戶的信息或者控制用戶的電腦。 舉一個通俗的例子,早期使用JSP頁面渲染頁面的項目,如果將用戶名改成nick<alert>1</alert>,則當用戶打開頁面時,就會彈出一個警告框,而這個警告框可以被惡意腳本所 ...
一周排行
    -Advertisement-
    Play Games
  • 基於.NET Framework 4.8 開發的深度學習模型部署測試平臺,提供了YOLO框架的主流系列模型,包括YOLOv8~v9,以及其系列下的Det、Seg、Pose、Obb、Cls等應用場景,同時支持圖像與視頻檢測。模型部署引擎使用的是OpenVINO™、TensorRT、ONNX runti... ...
  • 十年沉澱,重啟開發之路 十年前,我沉浸在開發的海洋中,每日與代碼為伍,與演算法共舞。那時的我,滿懷激情,對技術的追求近乎狂熱。然而,隨著歲月的流逝,生活的忙碌逐漸占據了我的大部分時間,讓我無暇顧及技術的沉澱與積累。 十年間,我經歷了職業生涯的起伏和變遷。從初出茅廬的菜鳥到逐漸嶄露頭角的開發者,我見證了 ...
  • C# 是一種簡單、現代、面向對象和類型安全的編程語言。.NET 是由 Microsoft 創建的開發平臺,平臺包含了語言規範、工具、運行,支持開發各種應用,如Web、移動、桌面等。.NET框架有多個實現,如.NET Framework、.NET Core(及後續的.NET 5+版本),以及社區版本M... ...
  • 前言 本文介紹瞭如何使用三菱提供的MX Component插件實現對三菱PLC軟元件數據的讀寫,記錄了使用電腦模擬,模擬PLC,直至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1. PLC開發編程環境GX Works2,GX Works2下載鏈接 https:// ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • 1、jQuery介紹 jQuery是什麼 jQuery是一個快速、簡潔的JavaScript框架,是繼Prototype之後又一個優秀的JavaScript代碼庫(或JavaScript框架)。jQuery設計的宗旨是“write Less,Do More”,即倡導寫更少的代碼,做更多的事情。它封裝 ...
  • 前言 之前的文章把js引擎(aardio封裝庫) 微軟開源的js引擎(ChakraCore))寫好了,這篇文章整點js代碼來測一下bug。測試網站:https://fanyi.youdao.com/index.html#/ 逆向思路 逆向思路可以看有道翻譯js逆向(MD5加密,AES加密)附完整源碼 ...
  • 引言 現代的操作系統(Windows,Linux,Mac OS)等都可以同時打開多個軟體(任務),這些軟體在我們的感知上是同時運行的,例如我們可以一邊瀏覽網頁,一邊聽音樂。而CPU執行代碼同一時間只能執行一條,但即使我們的電腦是單核CPU也可以同時運行多個任務,如下圖所示,這是因為我們的 CPU 的 ...
  • 掌握使用Python進行文本英文統計的基本方法,並瞭解如何進一步優化和擴展這些方法,以應對更複雜的文本分析任務。 ...
  • 背景 Redis多數據源常見的場景: 分區數據處理:當數據量增長時,單個Redis實例可能無法處理所有的數據。通過使用多個Redis數據源,可以將數據分區存儲在不同的實例中,使得數據處理更加高效。 多租戶應用程式:對於多租戶應用程式,每個租戶可以擁有自己的Redis數據源,以確保數據隔離和安全性。 ...