Java使用MD5对文件进行签名

使用MD5对文件进行加密获得签名串

package com.jani.test;

import java.io.File;

import java.io.FileInputStream;

import java.io.IOException;

import java.security.MessageDigest;

/**
 * Created by Jani on 2016/11/15.
 */
public class MD5Demo {

    /**
     * 对文件全文生成MD5摘要
     *
     * @param file
     * <p>
     * 要加密的文件
     * @return MD5摘要码
     */


    static char hexdigits[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8',

            '9', 'a', 'b', 'c', 'd', 'e', 'f'};


    public static String getMD5(File file) {


        FileInputStream fis = null;

        try {

            MessageDigest md = MessageDigest.getInstance("MD5");

            fis = new FileInputStream(file);

            byte[] buffer = new byte[2048];

            int length = -1;

            long s = System.currentTimeMillis();

            while ((length = fis.read(buffer)) != -1) {

                md.update(buffer, 0, length);

            }

            byte[] b = md.digest();

            return byteToHexString(b);

            // 16位加密

            // return buf.toString().substring(8, 24);

        } catch (Exception ex) {

            ex.printStackTrace();

            return null;

        } finally {

            try {

                fis.close();

            } catch (IOException ex) {

                ex.printStackTrace();

            }

        }

    }


    /** */

    /**
     * 把byte[]数组转换成十六进制字符串表示形式
     *
     * @param tmp 要转换的byte[]
     * @return 十六进制字符串表示形式
     */

    private static String byteToHexString(byte[] tmp) {

        String s;

        // 用字节表示就是 16 个字节

        char str[] = new char[16 * 2]; // 每个字节用 16 进制表示的话,使用两个字符,

        // 所以表示成 16 进制需要 32 个字符

        int k = 0; // 表示转换结果中对应的字符位置

        for (int i = 0; i < 16; i++) { // 从第一个字节开始,对 MD5 的每一个字节

            // 转换成 16 进制字符的转换

            byte byte0 = tmp[i]; // 取第 i 个字节

            str[k++] = hexdigits[byte0 >>> 4 & 0xf]; // 取字节中高 4 位的数字转换,

            // >>> 为逻辑右移,将符号位一起右移

            str[k++] = hexdigits[byte0 & 0xf]; // 取字节中低 4 位的数字转换

        }

        s = new String(str); // 换后的结果转换为字符串

        return s;

    }


    public static void main(String arg[]) {

        System.out.println(getMD5(new File("D:/5.apk")));

    }

}
  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
首先,您需要下载并安装 BC(Bouncy Castle)密码库,它提供了对SM2和SM3算法的支持。然后,您可以按照以下步骤使用SM2算法生成文件签名: 1. 读取待签名文件内容,并进行SHA-256哈希处理。 2. 生成SM2密钥对,包括公钥和私钥。 3. 使用私钥对哈希值进行数字签名。 4. 将签名结果和公钥一起写入文件中,形成签名文件。 下面是一个示例代码: ```java import java.io.*; import java.security.*; import org.bouncycastle.asn1.pkcs.PrivateKeyInfo; import org.bouncycastle.crypto.params.*; import org.bouncycastle.jcajce.provider.asymmetric.ec.BCECPrivateKey; import org.bouncycastle.jcajce.provider.asymmetric.ec.BCECPublicKey; import org.bouncycastle.jce.provider.BouncyCastleProvider; import org.bouncycastle.openssl.PEMParser; import org.bouncycastle.openssl.jcajce.JcaPEMKeyConverter; import org.bouncycastle.util.encoders.Hex; public class SM2Signature { public static void main(String[] args) throws Exception { Security.addProvider(new BouncyCastleProvider()); // 读取待签名文件 File file = new File("test.txt"); byte[] data = new byte[(int) file.length()]; InputStream in = new FileInputStream(file); in.read(data); in.close(); // 计算文件哈希值 MessageDigest md = MessageDigest.getInstance("SHA-256"); byte[] hash = md.digest(data); // 生成SM2密钥对 KeyPairGenerator kpg = KeyPairGenerator.getInstance("EC", "BC"); ECGenParameterSpec ecsp = new ECGenParameterSpec("sm2p256v1"); kpg.initialize(ecsp); KeyPair keyPair = kpg.generateKeyPair(); BCECPrivateKey privateKey = (BCECPrivateKey) keyPair.getPrivate(); BCECPublicKey publicKey = (BCECPublicKey) keyPair.getPublic(); // 对哈希值进行数字签名 Signature signature = Signature.getInstance("SM3withSM2", "BC"); signature.initSign(privateKey); signature.update(hash); byte[] sigBytes = signature.sign(); // 将签名结果和公钥写入文件 FileOutputStream out = new FileOutputStream("test.sig"); out.write(sigBytes); out.write(publicKey.getQ().getEncoded(false)); out.close(); } } ``` 您可以使用以下代码验证签名: ```java import java.io.*; import java.security.*; import org.bouncycastle.asn1.pkcs.PrivateKeyInfo; import org.bouncycastle.crypto.params.*; import org.bouncycastle.jcajce.provider.asymmetric.ec.BCECPrivateKey; import org.bouncycastle.jcajce.provider.asymmetric.ec.BCECPublicKey; import org.bouncycastle.jce.provider.BouncyCastleProvider; import org.bouncycastle.openssl.PEMParser; import org.bouncycastle.openssl.jcajce.JcaPEMKeyConverter; import org.bouncycastle.util.encoders.Hex; public class SM2SignatureVerify { public static void main(String[] args) throws Exception { Security.addProvider(new BouncyCastleProvider()); // 读取待验证的文件签名文件 File file = new File("test.txt"); byte[] data = new byte[(int) file.length()]; InputStream in = new FileInputStream(file); in.read(data); in.close(); File sigFile = new File("test.sig"); byte[] sigData = new byte[(int) sigFile.length()]; in = new FileInputStream(sigFile); in.read(sigData); in.close(); // 计算文件哈希值 MessageDigest md = MessageDigest.getInstance("SHA-256"); byte[] hash = md.digest(data); // 解析签名文件中的签名和公钥 byte[] sig = new byte[64]; byte[] pub = new byte[65]; System.arraycopy(sigData, 0, sig, 0, 64); System.arraycopy(sigData, 64, pub, 0, 65); X9ECParameters ecParams = ECNamedCurveTable.getByName("sm2p256v1"); ECPoint q = ecParams.getCurve().decodePoint(pub); ECPublicKeyParameters pubParams = new ECPublicKeyParameters(q, new ECNamedDomainParameters("sm2p256v1", ecParams.getCurve(), ecParams.getG(), ecParams.getN())); // 验证签名 Signature signature = Signature.getInstance("SM3withSM2", "BC"); signature.initVerify(pubParams); signature.update(hash); boolean result = signature.verify(sig); System.out.println(result); } } ``` 需要注意的是,验证签名时需要使用签名文件中的公钥进行验证。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值