Java使用ElGamal算法及需要注意的几点问题

2023-12-09 19:38

本文主要是介绍Java使用ElGamal算法及需要注意的几点问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近在搞一个使用ElGamal算法的demo,遇到了不少大坑,记录下来。

关于ElGamal的使用,参考该链接:非对称加密算法-ElGamal算法

他这个使用,由于jdk并没有实现ElGamal,所以得引用两个jar包:bcprov-jdk16-145.jar 和commons-codec-1.11.jar,而这也成为了后来的万恶之源。

原链接的前辈写的代码是这样的:

//初始化密钥  //生成密钥对  Map<String,Object> keyMap=ElGamalCoder.initKey();  //公钥  byte[] publicKey=ElGamalCoder.getPublicKey(keyMap);  //私钥  byte[] privateKey=ElGamalCoder.getPrivateKey(keyMap);  System.out.println("公钥:/n"+Base64.encodeBase64String(publicKey));  System.out.println("私钥:/n"+Base64.encodeBase64String(privateKey));  System.out.println("================密钥对构造完毕,甲方将公钥公布给乙方,开始进行加密数据的传输=============");  String str="ElGamal密码交换算法";  System.out.println("/n===========甲方向乙方发送加密数据==============");  System.out.println("原文:"+str);  //乙方使用公钥对数据进行加密  byte[] code2=ElGamalCoder.encryptByPublicKey(str.getBytes(), publicKey);  System.out.println("===========乙方使用公钥对数据进行加密==============");  System.out.println("加密后的数据:"+Base64.encodeBase64String(code2));  //甲方使用私钥对数据进行解密  byte[] decode2=ElGamalCoder.decryptByPrivateKey(code2, privateKey);  System.out.println("甲方解密后的数据:"+new String(decode2));

密钥生成、加解密都在一个函数里面,这样子运行的时候没问题,也不存在乱码,因为是直接用的byte[]数组作为明文和密文进行加解密。但是,我的需求是需要在不同的地方进行公私钥的生成、加密、解密,于是就出现了两个问题:

1.不将byte数组作为参数,而是将byte数组序列化为字符串,将字符串作为密钥传入进行加密解密的时候会出现解密结果是乱码的问题。

解决办法:我没有找到合适的序列化的方法将生成的byte[]格式的公私钥序列化为正确的字符串,最后使用文件的方式解决:将byte[]用字节流写入文件,生成两个文件:publicKey和privateKey,使用的时候将文件中的二进制数据读取为byte[],使用这个byte[]作为参数传入,就不会产生乱码的问题。

2.我写了三个函数将密钥生成、加密和解密分开,单独调用的时候报如下异常:

Exception in thread "main" java.security.NoSuchAlgorithmException: ElGamal KeyFactory not availableat java.security.KeyFactory.<init>(Unknown Source)at java.security.KeyFactory.getInstance(Unknown Source)at com.sta.Util.ElGamalCoder.encryptByPublicKey(ElGamalCoder.java:104)at com.sta.Util.ElGamalCoder.decrypt(ElGamalCoder.java:211)at com.sta.Util.ElGamalCoder.main(ElGamalCoder.java:249)
我们来看一下这句话:
ElGamal KeyFactory not available

这句话产生的原因就是上面说的,JDK并没有自己实现ElGamal算法,需要添加额外的支持,链接中的前辈写的代码由于是密钥生成、加解密都在一个函数中调用,因此在密钥生成时写的添加ElGamal支持的代码对加解密方法同样生效,但是将加解密和密钥生成分开,分别调用时,KeyFactory就会没有ElGamal的支持,因此抛出上面的异常,具体细节可参考该链接: java中的秘钥工厂KeyFactory

所以我将前辈写的代码做了如下改动:

/*** 公钥加密* @param data待加密数据* @param key 密钥* @return byte[] 加密数据* */public static byte[] encryptByPublicKey(byte[] data,byte[] key) throws Exception{//------------------以下是改动,添加ElGamal支持------------------------------------------------//加入对BouncyCastle支持Security.addProvider(new BouncyCastleProvider());AlgorithmParameterGenerator apg=AlgorithmParameterGenerator.getInstance(KEY_ALGORITHM);//初始化参数生成器apg.init(KEY_SIZE);//生成算法参数AlgorithmParameters params=apg.generateParameters();//构建参数材料DHParameterSpec elParams=(DHParameterSpec)params.getParameterSpec(DHParameterSpec.class);//--------------------------------------------------------------------------------------------//实例化密钥工厂KeyFactory keyFactory=KeyFactory.getInstance(KEY_ALGORITHM);//初始化公钥//密钥材料转换X509EncodedKeySpec x509KeySpec=new X509EncodedKeySpec(key);//产生公钥PublicKey pubKey=keyFactory.generatePublic(x509KeySpec);//数据加密Cipher cipher=Cipher.getInstance(keyFactory.getAlgorithm());cipher.init(Cipher.ENCRYPT_MODE, pubKey);return cipher.doFinal(data);}

解密方法同理,添加上ElGamal支持的代码即可。

另外代码中密钥长度为256位,在某些情况下密钥长度超过128

位会报如下异常:NoSuchAlgorithmException,这个问题是美

帝的出口限制导致的,解决办法如图:


最后附我修改之后的ElGamalCoder.java:  该class通过调用ElGamalCoder.getKeyFile()方法生成公钥和私钥,公钥文件为publicKey,私钥文件为privateKey。

加密:String miwen = ElGamalCoder.decrypt("这里传入明文", "publicKey"); //publicKey是公钥文件所在的路径(包括文件名)

解密:String mingwen = ElGamalCoder.encrypt("这里传入加密之后的密文","privateKey");//privateKey是私钥文件所在路径

java文件下载:ElGamalCoder.java

-end-

这篇关于Java使用ElGamal算法及需要注意的几点问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/474843

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置