82. 枚举

枚举类是一种很实用的类型,在标识状态一类的应用中,使用起来十分方便。当然我们也可以用map来替代枚举类型的功能,虽然简单,但是阅读起来map要比枚举稍微差一些,可以根据情况来选择。

map

比较清晰的直接赋值为一个map形式。

package github.banana;

import java.util.HashMap;
import java.util.Map;

/**
 * 供应商类型和描述
 */
public class TP {

    public static String getName(String id) {
        Map<String, String> map = new HashMap<>();
        map.put("1", "携程");
        map.put("2", "国航");
        map.put("4", "东航");
        map.put("8", "南航");
        map.put("16", "携程一期H5");
        map.put("32", "必去");
        map.put("64", "携程H5抓取");
        map.put("128", "必去国际");
        map.put("256", "就旅行");
        map.put("512", "必去候补");
        map.put("1024", "去哪儿");
        map.put("2048", "携程H5阿拉丁");
        map.put("ctrip", "携程");

        return map.getOrDefault(id, "未知");
    }
}

使用枚举来实现。

package github.banana;

/**
 * 供应商类型和描述
 */
public enum  EnumTP {
    CTRIP("1", "携程"),
    CA("2", "国航"),
    MU("4", "东航"),
    CZ("8", "南航"),
    CTRIPH5("16", "携程一期H5"),
    BIQU("32", "必去"),
    CTRIPCRAWL("64", "携程H5抓取"),
    BIQUINT("128", "必去国际"),
    BIQUSTANDARD("256", "就旅行"),
    QUNAR("512", "去哪儿"),
    CTRIPHTALADDIN("1024", "携程H5阿拉丁"),
    CTRIPV2("ctrip", "携程");

    // 供应商标识
    private String index;

    // 供应商描述
    private String name;

    EnumTP(String index, String name) {
        this.index = index;
        this.name = name;
    }

    // 获取供应商中文名称
    public static String getName(String index) {
        for (EnumTP enumTP:EnumTP.values()) {
            if (index.equals(enumTP.index)) {
                return enumTP.name;
            }
        }
        return "未知";
    }
}

可以看出,如果操作本身比较简单,其实选择哪一种都是比较合乎逻辑的。但就是设计来说,map因数据结构的关系,理论上会更快一些,如果做持久化后。不过这些性能,可能并不明显。

package github.banana;

public class TPTest {

    public static void main(String[] args) {
        String partner = "256";

        System.out.println(TP.getName(partner));

        System.out.println(EnumTP.getName(partner));
    }
}

Read More

81. 解密微信小程序对用户加密的数据

微信小程序通过微信获取用户信息接口wx.getUserInfo返回用户的openId为加密数据,需要交由对应的服务端来进行解密。微信小程序提供的示例代码中没有Java版本,刚好我自己的服务端需要用Java实现,故补充Java版本的代码实例,其它版本的代码示例请前往自行下载研究。

package com.github.zhgxun.test.controller;

import javax.crypto.BadPaddingException;
import javax.crypto.Cipher;
import javax.crypto.IllegalBlockSizeException;
import javax.crypto.NoSuchPaddingException;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import java.io.UnsupportedEncodingException;
import java.security.InvalidAlgorithmParameterException;
import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException;
import java.util.Base64;

/**
 * 对微信小程序用户加密数据进行解密
 * AES是高级加密标准(Advanced Encryption Standard)的简称
 *
 * @link https://mp.weixin.qq.com/debug/wxadoc/dev/api/signature.html#wxchecksessionobject
 * @link https://magiclen.org/java-base64/ Java如何進行Base64的編碼(Encode)與解碼(Decode)
 */
public class AesTest {

    /**
     * jdk 8开始内置的Base64相关操作做过优化, 性能较高, 不需要使用第三方库
     */
    private final static Base64.Decoder decoder = Base64.getDecoder();

    public static void main(String args[]) {

        // 应用ID
        String appId = "wx4f4bc4dec97d474b";
        // 安全字符串
        String sessionKey = "tiihtNczf5v6AKRyjwEUhQ==";
        // 加密后的字符串
        String encrypt = "CiyLU1Aw2KjvrjMdj8YKliAjtP4gsMZMQmRzooG2xrDcvSnxIMXFufNstNGTyaGS9uT5geRa0W4oTOb1WT7fJlAC+oNPdbB+3hVbJSRgv+4lGOETKUQz6OYStslQ142dNCuabNPGBzlooOmB231qMM85d2/fV6ChevvXvQP8Hkue1poOFtnEtpyxVLW1zAo6/1Xx1COxFvrc2d7UL/lmHInNlxuacJXwu0fjpXfz/YqYzBIBzD6WUfTIF9GRHpOn/Hz7saL8xz+W//FRAUid1OksQaQx4CMs8LOddcQhULW4ucetDf96JcR3g0gfRK4PC7E/r7Z6xNrXd2UIeorGj5Ef7b1pJAYB6Y5anaHqZ9J6nKEBvB4DnNLIVWSgARns/8wR2SiRS7MNACwTyrGvt9ts8p12PKFdlqYTopNHR1Vf7XjfhQlVsAJdNiKdYmYVoKlaRv85IfVunYzO0IKXsyl7JCUjCpoG20f0a04COwfneQAGGwd5oa+T8yO5hzuyDb/XcxxmK01EpqOyuxINew==";
        // 初始向量Iv(Initialization Vector), 使用除ECB以外的其他加密模式均需要传入一个初始向量, 其大小与块大小相等, AES块大小是128bit, 所以Iv的长度是16字节, 初始向量可以加强算法强度
        String iv = "r7BXXKkLb8qrSNn05n0qiA==";

        if (sessionKey.length() != 24) {
            System.out.println("encodingAesKey 非法");
        }
        if (iv.length() != 24) {
            System.out.println("iv非法");
        }

        try {
            // 填充方式(Padding)决定了最后的一个块需要填充的内容, 填充方式有PKCS5Padding, PKCS7Padding, NOPADDING三种, 但是JDK只提供了PKCS5Padding, NOPADDING两种, 填充方式为PKCS5Padding时, 最后一个块需要填充X个字节,填充的值就是X; 填充方式为NOPADDING时, 最后的一个块填充的内容由程序员自己决定, 通常填充0
            // @link https://docs.oracle.com/javase/8/docs/api/index.html 支持的模式
            Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
            cipher.init(Cipher.DECRYPT_MODE, new SecretKeySpec(decoder.decode(sessionKey), "AES"), new IvParameterSpec(decoder.decode(iv)));
            byte[] bytes = cipher.doFinal(decoder.decode(encrypt.getBytes("UTF-8")));
            System.out.println(new String(bytes));
        } catch (NoSuchPaddingException | NoSuchAlgorithmException | UnsupportedEncodingException | IllegalBlockSizeException | BadPaddingException | InvalidKeyException | InvalidAlgorithmParameterException e) {
            e.printStackTrace();
        }
    }
}