SM4 国密算法_sm4/ecb/pkcs5padding_zhglhy的博客-程序员宅基地

技术标签: java  网络  网络协议  p2p  

public class Sm4 {


    static {
        Security.addProvider(new BouncyCastleProvider());
    }

    private static final String ENCODING = "UTF-8";
    public static final String ALGORITHM_NAME = "SM4";
    // 加密算法/分组加密模式/分组填充方式
    // PKCS5Padding-以8个字节为一组进行分组加密
    // 定义分组加密模式使用:PKCS5Padding
    public static final String ALGORITHM_NAME_ECB_PADDING = "SM4/ECB/PKCS5Padding";
    // 128-32位16进制;256-64位16进制
    public static final int DEFAULT_KEY_SIZE = 128;

    /**
     * 自动生成密钥
     *
     * @return
     * @explain
     */
    public static String generateKey() throws Exception {
        return new String(Hex.encodeHex(generateKey(DEFAULT_KEY_SIZE),false));
    }

    /**
     * @param keySize
     * @return
     * @throws Exception
     * @explain
     */
    public static byte[] generateKey(int keySize) throws Exception {
        KeyGenerator kg = KeyGenerator.getInstance(ALGORITHM_NAME, BouncyCastleProvider.PROVIDER_NAME);
        kg.init(keySize, new SecureRandom());
        return kg.generateKey().getEncoded();
    }

    /**
     * 生成ECB暗号
     *
     * @param algorithmName 算法名称
     * @param mode          模式
     * @param key
     * @return
     * @throws Exception
     * @explain ECB模式(电子密码本模式:Electronic codebook)
     */
    private static Cipher generateEcbCipher(String algorithmName, int mode, byte[] key) throws Exception {
        Cipher cipher = Cipher.getInstance(algorithmName, BouncyCastleProvider.PROVIDER_NAME);
        Key sm4Key = new SecretKeySpec(key, ALGORITHM_NAME);
        cipher.init(mode, sm4Key);
        return cipher;
    }

    /**
     * sm4加密
     *
     * @param hexKey   16进制密钥(忽略大小写)
     * @param paramStr 待加密字符串
     * @return 返回16进制的加密字符串
     * @explain 加密模式:ECB
     * 密文长度不固定,会随着被加密字符串长度的变化而变化
     */
    public static String encryptEcb(String hexKey, String paramStr) {
        try {
            String cipherText = "";
            // 16进制字符串-->byte[]
            byte[] keyData = ByteUtils.fromHexString(hexKey);
            // String-->byte[]
            byte[] srcData = paramStr.getBytes(ENCODING);
            // 加密后的数组
            byte[] cipherArray = encrypt_Ecb_Padding(keyData, srcData);
            // byte[]-->hexString
            cipherText = ByteUtils.toHexString(cipherArray);
            return cipherText;
        } catch (Exception e) {
            return paramStr;
        }
    }

    /**
     * 加密模式之Ecb
     *
     * @param key
     * @param data
     * @return
     * @throws Exception
     * @explain
     */
    public static byte[] encrypt_Ecb_Padding(byte[] key, byte[] data) throws Exception {
        Cipher cipher = generateEcbCipher(ALGORITHM_NAME_ECB_PADDING, Cipher.ENCRYPT_MODE, key);
        return cipher.doFinal(data);
    }

    /**
     * sm4解密
     *
     * @param hexKey     16进制密钥
     * @param cipherText 16进制的加密字符串(忽略大小写)
     * @return 解密后的字符串
     * @throws Exception
     * @explain 解密模式:采用ECB
     */
    public static String decryptEcb(String hexKey, String cipherText) {
        // 用于接收解密后的字符串
        String decryptStr = "";
        // hexString-->byte[]
        byte[] keyData = ByteUtils.fromHexString(hexKey);
        // hexString-->byte[]
        byte[] cipherData = ByteUtils.fromHexString(cipherText);
        // 解密
        byte[] srcData = new byte[0];
        try {
            srcData = decrypt_Ecb_Padding(keyData, cipherData);
            // byte[]-->String
            decryptStr = new String(srcData, ENCODING);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return decryptStr;
    }

    /**
     * 解密
     *
     * @param key
     * @param cipherText
     * @return
     * @throws Exception
     * @explain
     */
    public static byte[] decrypt_Ecb_Padding(byte[] key, byte[] cipherText) throws Exception {
        Cipher cipher = generateEcbCipher(ALGORITHM_NAME_ECB_PADDING, Cipher.DECRYPT_MODE, key);
        return cipher.doFinal(cipherText);
    }

    /**
     * 校验加密前后的字符串是否为同一数据
     *
     * @param hexKey     16进制密钥(忽略大小写)
     * @param cipherText 16进制加密后的字符串
     * @param paramStr   加密前的字符串
     * @return 是否为同一数据
     * @throws Exception
     * @explain
     */
    public static boolean verifyEcb(String hexKey, String cipherText, String paramStr) throws Exception {
        // 用于接收校验结果
        boolean flag = false;
        // hexString-->byte[]
        byte[] keyData = ByteUtils.fromHexString(hexKey);
        // 将16进制字符串转换成数组
        byte[] cipherData = ByteUtils.fromHexString(cipherText);
        // 解密
        byte[] decryptData = decrypt_Ecb_Padding(keyData, cipherData);
        // 将原字符串转换成byte[]
        byte[] srcData = paramStr.getBytes(ENCODING);
        // 判断2个数组是否一致
        flag = Arrays.equals(decryptData, srcData);
        return flag;
    }


    public static void main(String[] args) {

        String key = "74d32aad624c62be6f774648530af79e";

        String cipher = "hello world";

        try {

            String enStr = Sm4Util.encryptEcb(key, cipher);
            System.out.println(enStr);

            String json = Sm4Util.decryptEcb(key, enStr);
            System.out.println(json);


        } catch (Exception exception) {
            exception.printStackTrace();
        }

    }


}
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/zhglhy/article/details/122961968

智能推荐

腾讯七大事业群简介-程序员宅基地

腾讯在现在把公司的组织架构分成七大事业群,其实有点类似于七个相对独立的公司一样,只不过事业群之间还是 共享很多资源,也有很多关联; 七个事业群分别是: TEG技术工程事业群、CDG企业发展事业群、IEG互动娱乐事业群、SNG社交网络事业群、WXG微信事业群、OMG网络媒体事业群、MIG移动互联网事业群1.TEG技术工程事业群 没有直接的商业产品,是其他事业群的支撑平台2.CDG企业发展事业群_事业群

oracle 的ROWNUM 和 ROW_NUMBER,以及分组排序_oracle 根据字段rownum用法排序-程序员宅基地

Oracle ROWNUM vs Oracle ROW_NUMBERROWNUM 是伪列,并没有实际意义。在ORDER BY执行之前,oracle给每一行查询结果赋予一个编号。ROW_NUMBER 是oracle提供的函数,通过参数为查询结果编号。区别ROWNUM 是伪列,没有参数。ROW_NUMBER 是分析函数,有参数。ROWNUM 在ORDER BY之前赋予查询结果。ROW_NUMBER 是 column 计算的一部分。ROWNUM 是唯一的。ROW_NUMBER 可以重复。代码例子_oracle 根据字段rownum用法排序

树形dp+dp总结(待续)_dp++-程序员宅基地

最近学了一下树形dp。主要做过:树形背包,树上路径,点覆盖,最大独立集,支配集这几类树形dp的题。总的来说,树形dp因为是树,子问题特别明显,可以在dfs时直接考虑如何转移,具体题目还是具体分析。有时转移会具有后效性,但可以通过其他方式消除(支配集就是一个很好的例子),有关树形dp的一些博客题解放在了前面,这篇就到这吧。最后心得:其实关于dp都是这样,一般从状态或者从决策入手,有时也可以从..._dp++

程序员终极武器_hulft utlsend -sync-程序员宅基地

********************************LInux********************************1,Tera Termhttp://www.forest.impress.co.jp/lib/inet/servernt/remote/utf8teraterm.html**************************_hulft utlsend -sync

Unique Snowflakes(JSU-ZJJ)-程序员宅基地

题目描述Emily the entrepreneur has a cool business idea: packaging and selling snowflakes. She has devised a machine that captures snowflakes as they fall, and serializes them into a stream of snowflake...

Android开发——长按触发事件_setontouchlistener 长按-程序员宅基地

1,绑定控件,设置触发事件View gain_set_sub_6 = findViewById(R.id.gain_set_sub_6);gain_set_sub_6.setOnTouchListener((v, event) -> { if(event.getAction() == MotionEvent.ACTION_DOWN){ updateAddOrSubtract(v.getId()); //手指按下时触发不停的发送消息 }else if(eve_setontouchlistener 长按

随便推点

CAS无锁机制原理-程序员宅基地

CAS无锁机制原理原子类java.util.concurrent.atomic包:原子类的小工具包,支持在单个变量上解除锁的线程安全编程原子变量类相当于一种泛化的 volatile 变量,能够支持原子的和有条件的读-改-写操作。AtomicInteger 表示一个int类型的值,并提供了 get 和 set 方法,这些 Volatile 类型的int变量在读取和写入上有着相同的内存语义..._cas是无锁操做

win10下的使用dlib的GPU加速_windows dlib开启gpu-程序员宅基地

发现上文已经配置好了dlib的环境,默认是有GPU加速功能的。在cmake的时候有显示 Enabling CUDA support for dlib. DLIB WILL USE CUDA之类的字眼,就是支持GUP加速。本地已经安装了1.CUDA 9.02.显卡是P20003.下载CUDNN,将解压出来的cuda文件放在duda9.0下。 将cuda/bin和cud..._windows dlib开启gpu

深度前馈网络与Xavier初始化原理_bp xavier-程序员宅基地

前言基本的神经网络的知识(一般化模型、前向计算、反向传播及其本质、激活函数等)小夕已经介绍完毕,本文先讲一下深度前馈网络的BP过程,再基于此来重点讲解在前馈网络中用来初始化model参数的Xavier方法的原理。前向前向过程很简单,每一层与下一层的连接边(即参数)构成了一个矩阵(即线性映射),每一层的神经元构成一个激活函数阵列(即非线性映射),信号便从输入层开始反复的重复这两个过程直到..._bp xavier

Enhanced Deep Residual Networks for Single Image Super-Resolution NTIRE2017(EDSR)_edsr损失函数-程序员宅基地

1 EDSR1.1 相关研究VDSR(CVPR2016)和SRResNet(CVPR2017)1.跳跃链接,全局和局部的跳跃链接使模型更深更稳固。2.尺寸放大方法,使用子像素卷积的后放大尺寸比之前放大更有效。但,可能被限制在单尺寸超分辨率重建。1.2 EDSR的四个改进点1.batch-normalization不像分类问题,输入和输出有相似的分布。在SR中,可能不希望标准化中..._edsr损失函数

一位不错领导说的话-程序员宅基地

为什么80%的码农都做不了架构师?>>> ..._占用朋友圈的空间怎么写感谢的话

【SQL】运用JDBC实现一个注册、登录系统的编写-程序员宅基地

数据库的建立首先,建立一个数据库,存储注册成功的账户信息。其SQL的DDL语句如下:CREATE TABLE `jdbctest` ( `id` int(10) NOT NULL auto_increment, `username` varchar(20) default NULL, `password` varchar(20) default NULL, `realN