Java SM2 国密算法使用

Java SM2 国密算法使用

文章目录

  • Java SM2 国密算法使用
    • 生成密钥对
    • 加载HEX密钥对字符串
    • 签名和验签
    • 加解密

这里需要引入对应的依赖:

<dependency>
  <groupId>org.bouncycastlegroupId>
  <artifactId>bcprov-jdk15onartifactId>
  <version>1.65version>
dependency>

生成密钥对

BouncyCastleProvider provider = new BouncyCastleProvider();
// 获取椭圆曲线相关生成参数规格
ECGenParameterSpec genParameterSpec = new ECGenParameterSpec("sm2p256v1");
// 获取一个椭圆曲线类型的密钥对生成器
KeyPairGenerator keyPairGenerator = KeyPairGenerator.getInstance("EC", provider);
// 使用SM2的算法区域初始化密钥生成器
keyPairGenerator.initialize(genParameterSpec, new SecureRandom());

// 生成密钥对
KeyPair keyPair = keyPairGenerator.generateKeyPair();
BCECPrivateKey privateKey = (BCECPrivateKey) keyPair.getPrivate();
BCECPublicKey publicKey = (BCECPublicKey) keyPair.getPublic();

// 拿到32字节的私钥HEX
System.out.println("Private Key: " + privateKey.getD().toString(16));

// true  代表压缩密钥,以02、03开头,长度为33字节
// false 代表未压缩,以04开头,长度为65字节
System.out.println("Public Key: " + Hex.encode(publicKey.getQ().getEncoded(true)));

加载HEX密钥对字符串

String prvKey = "私钥HEX字符串", pubKey = "公钥HEX字符串";
// 获取SM2相关参数
X9ECParameters parameters = GMNamedCurves.getByName("sm2p256v1");
// 椭圆曲线参数规格
ECParameterSpec ecParameterSpec = new ECParameterSpec(parameters.getCurve(), parameters.getG(), parameters.getN(), parameters.getH());
// 将公钥HEX字符串转换为椭圆曲线对应的点
ECPoint ecPoint = parameters.getCurve().decodePoint(Hex.decode(pubKey));
// 将私钥HEX字符串转换为X值
BigInteger bigInteger = new BigInteger(prvKey, 16);
// 获取椭圆曲线KEY生成器
KeyFactory keyFactory = KeyFactory.getInstance("EC", provider);
// 将X值转为私钥KEY对象
BCECPrivateKey privateKey = (BCECPrivateKey) keyFactory.generatePrivate(new ECPrivateKeySpec(bigInteger, ecParameterSpec));
// 将椭圆曲线点转为公钥KEY对象
BCECPublicKey publicKey = (BCECPublicKey) keyFactory.generatePublic(new ECPublicKeySpec(ecPoint, ecParameterSpec));

签名和验签

byte[] bytes = "Hello World".getBytes(), signBytes;
// 创建签名对象
Signature signature = Signature.getInstance(GMObjectIdentifiers.sm2sign_with_sm3.toString(), provider);
// 初始化为签名状态
signature.initSign(privateKey);
// 传入签名字节
signature.update(bytes);
// 返回签名字节
System.out.println(Hex.encode(signBytes = signature.sign()));
// 初始化为验签状态
signature.initVerify(publicKey);
// 传入签名字节
signature.update(bytes);
// 返回验签结果
System.out.println(signature.verify(signBytes));

加解密

注:这里加解密与RSA有所区别,加密要用公钥加密,用私钥解密

String word = "你好世界";
// 获取SM2加密器
Cipher cipher = Cipher.getInstance("SM2", provider);
// 初始化为加密模式
cipher.init(Cipher.ENCRYPT_MODE, publicKey);
// 加密并编码为base64格式
word = Base64.getEncoder().encodeToString(cipher.doFinal(word.getBytes()));
System.out.println("密文:" + word);
// 初始化为解密模式
cipher.init(Cipher.DECRYPT_MODE, privateKey);
// 解密
word = new String(cipher.doFinal(Base64.getDecoder().decode(word)));
System.out.println("解密:" + word);

以上就是全部内容。

你可能感兴趣的:(Java,加密解密,SM2,SM3,SM4,国密算法,国家标准规范)