非對稱算法DM

前言

    最近公司的業務需要對部分字段加密後傳輸,學習了一些加密算法相關的知識,加密種類大致分三類:

     1.對稱加密:公鑰與私鑰相同

     2.非對稱加密:公鑰與私鑰不同,公鑰暴露,私鑰保密

     3.哈希加密:不可逆加密,多用來判斷

    下面簡單介紹下常用的非對稱性加密:DM加密

package com.jm;
import sun.misc.BASE64Decoder;
import sun.misc.BASE64Encoder;
import javax.crypto.Cipher;
import javax.crypto.KeyAgreement;
import javax.crypto.SecretKey;
import javax.crypto.interfaces.DHPrivateKey;
import javax.crypto.interfaces.DHPublicKey;
import javax.crypto.spec.DHParameterSpec;
import java.security.*;
import java.security.spec.PKCS8EncodedKeySpec;
import java.security.spec.X509EncodedKeySpec;
import java.util.HashMap;
import java.util.Map;
/**
 * Diffie- Hellman算法(D-H算法),密鑰一致協議。
 * 是由公開密鑰密碼體制的奠基人Diffie和Hellman所提出的一種思想。
 * 簡單的說就是允許兩名用 戶在公開媒體上交換信息以生成"一致"的、可以共享的密鑰。
 * 換句話說,就是由甲方產出一對密鑰(公鑰、私鑰),乙方依照甲方公鑰產生乙方密鑰對(公鑰、私 鑰)。
 * 以此爲基線,作爲數據傳輸保密基礎,同時雙方使用同一種對稱加密算法構建本地密鑰(SecretKey)對數據加密。
 * 這樣,在互通了本地密鑰 (SecretKey)算法後,甲乙雙方公開自己的公鑰,使用對方的公鑰和剛纔產生的私鑰加密數據,
 * 同時可以使用對方的公鑰和自己的私鑰對數據解密。
 * 不單 單是甲乙雙方兩方,可以擴展爲多方共享數據通訊,這樣就完成了網絡交互數據的安全通訊!該算法源於中國的同餘定理——中國餘數定理。
 流程分析:
 1.甲方構建密鑰對兒,將公鑰公佈給乙方,將私鑰保留;雙方約定數據加密算法;乙方通過甲方公鑰構建密鑰對兒,將公鑰公佈給甲方,將私鑰保留。
 2.甲方使用私鑰、乙方公鑰、約定數據加密算法構建本地密鑰,然後通過本地密鑰加密數據,發送給乙方加密後的數據;
 乙方使用私鑰、甲方公鑰、約定數據加密算法構建本地密鑰,然後通過本地密鑰對數據解密。
 3.乙方使用私鑰、甲方公鑰、約定數據加密算法構建本地密鑰,然後通過本地密鑰加密數據,發送給甲方加密後的數據;
 甲方使用私鑰、乙方公鑰、約定數據加密算法構建本地密鑰,然後通過本地密鑰對數據解密。
 */
public class DMdemo {
    public static final String ALGORITHM = "DH";
    /**
     * 默認密鑰字節數
     *
     *
     * DH
     * Default Keysize 1024
     * Keysize must be a multiple of 64, ranging from 512 to 1024 (inclusive).
     *
     */
    private static final int KEY_SIZE = 1024;
    /**
     * DH加密下需要一種對稱加密算法對數據加密,這裏我們使用DES,也可以使用其他對稱加密算法。
     */
    public static final String SECRET_ALGORITHM = "DES";
    private static final String PUBLIC_KEY = "DHPublicKey";
    private static final String PRIVATE_KEY = "DHPrivateKey";
    /**
     * 初始化甲方密鑰
     *
     * @return
     * @throws Exception
     */
    public static Map initKey() throws Exception {
        KeyPairGenerator keyPairGenerator = KeyPairGenerator.getInstance(ALGORITHM);
        keyPairGenerator.initialize(KEY_SIZE);
        KeyPair keyPair = keyPairGenerator.generateKeyPair();
// 甲方公鑰
        DHPublicKey publicKey = (DHPublicKey) keyPair.getPublic();
// 甲方私鑰
        DHPrivateKey privateKey = (DHPrivateKey) keyPair.getPrivate();
        Map keyMap = new HashMap (2);
        keyMap.put(PUBLIC_KEY, publicKey);
        keyMap.put(PRIVATE_KEY, privateKey);
        return keyMap;
    }
    /**
     * 初始化乙方密鑰
     *
     * @param key 甲方公鑰
     * @return
     * @throws Exception
     */
    public static Map initKey(String key) throws Exception {
// 解析甲方公鑰
        byte[] keyBytes = new BASE64Decoder().decodeBuffer(key);
        X509EncodedKeySpec x509KeySpec = new X509EncodedKeySpec(keyBytes);
        KeyFactory keyFactory = KeyFactory.getInstance(ALGORITHM);
        PublicKey pubKey = keyFactory.generatePublic(x509KeySpec);
// 由甲方公鑰構建乙方密鑰
        DHParameterSpec dhParamSpec = ((DHPublicKey) pubKey).getParams();
        KeyPairGenerator keyPairGenerator = KeyPairGenerator.getInstance(keyFactory.getAlgorithm());
        keyPairGenerator.initialize(dhParamSpec);
        KeyPair keyPair = keyPairGenerator.generateKeyPair();
// 乙方公鑰
        DHPublicKey publicKey = (DHPublicKey) keyPair.getPublic();
// 乙方私鑰
        DHPrivateKey privateKey = (DHPrivateKey) keyPair.getPrivate();
        Map keyMap = new HashMap (2);
        keyMap.put(PUBLIC_KEY, publicKey);
        keyMap.put(PRIVATE_KEY, privateKey);
        return keyMap;
    }
    /**
     * 加密
     *
     * @param data
     * 待加密數據
     * @param publicKey
     * 甲方公鑰
     * @param privateKey
     * 乙方私鑰
     * @return
     * @throws Exception
     */
    public static String encrypt(String data, String publicKey, String privateKey) throws Exception {
// 生成本地密鑰
        SecretKey secretKey = getSecretKey(publicKey, privateKey);
// 數據加密
        Cipher cipher = Cipher.getInstance(secretKey.getAlgorithm());
        cipher.init(Cipher.ENCRYPT_MODE, secretKey);
        return new BASE64Encoder().encode(cipher.doFinal(data.getBytes("UTF-8")));
    }
    /**
     * 解密
     *
     * @param data
     * 待解密數據
     * @param publicKey
     * 乙方公鑰
     * @param privateKey
     * 乙方私鑰
     * @return
     * @throws Exception
     */
    public static String decrypt(String data, String publicKey, String privateKey) throws Exception {
// 生成本地密鑰
        SecretKey secretKey = getSecretKey(publicKey, privateKey);
// 數據解密
        Cipher cipher = Cipher.getInstance(secretKey.getAlgorithm());
        cipher.init(Cipher.DECRYPT_MODE, secretKey);
        byte[] decode_data = new BASE64Decoder().decodeBuffer(data);
        return new String(cipher.doFinal(decode_data));
    }
    /**
     * 構建密鑰
     *
     * @param publicKey
     * 公鑰
     * @param privateKey
     * 私鑰
     * @return
     * @throws Exception
     */
    private static SecretKey getSecretKey(String publicKey, String privateKey) throws Exception {
// 初始化公鑰
        byte[] pubKeyBytes = new BASE64Decoder().decodeBuffer(publicKey);
        KeyFactory keyFactory = KeyFactory.getInstance(ALGORITHM);
        X509EncodedKeySpec x509KeySpec = new X509EncodedKeySpec(pubKeyBytes);
        PublicKey pubKey = keyFactory.generatePublic(x509KeySpec);
// 初始化私鑰
        byte[] priKeyBytes = new BASE64Decoder().decodeBuffer(privateKey);
        PKCS8EncodedKeySpec pkcs8KeySpec = new PKCS8EncodedKeySpec(priKeyBytes);
        Key priKey = keyFactory.generatePrivate(pkcs8KeySpec);
        KeyAgreement keyAgree = KeyAgreement.getInstance(keyFactory.getAlgorithm());
        keyAgree.init(priKey);
        keyAgree.doPhase(pubKey, true);
// 生成本地密鑰
        SecretKey secretKey = keyAgree.generateSecret(SECRET_ALGORITHM);
        return secretKey;
    }
    /**
     * 取得私鑰
     *
     * @param keyMap
     * @return
     * @throws Exception
     */
    public static String getPrivateKey(Map keyMap) throws Exception {
        Key key = (Key) keyMap.get(PRIVATE_KEY);
        return new BASE64Encoder().encode(key.getEncoded());
    }
    /**
     * 取得公鑰
     *
     * @param keyMap
     * @return
     * @throws Exception
     */
    public static String getPublicKey(Map keyMap) throws Exception {
        Key key = (Key) keyMap.get(PUBLIC_KEY);
        return new BASE64Encoder().encode(key.getEncoded());
    }
    public static void main(String[] args) throws Exception{
// 生成甲方密鑰對兒
        Map aKeyMap = DMdemo.initKey();
        String aPublicKey = DMdemo.getPublicKey(aKeyMap);
        String aPrivateKey = DMdemo.getPrivateKey(aKeyMap);
        System.err.println("甲方公鑰:" + aPublicKey);
        System.err.println("甲方私鑰:" + aPrivateKey);
// 由甲方公鑰產生本地(乙方)密鑰對兒
        Map bKeyMap = DMdemo.initKey(aPublicKey);
        String bPublicKey = DMdemo.getPublicKey(bKeyMap);
        String bPrivateKey = DMdemo.getPrivateKey(bKeyMap);
        System.err.println("乙方公鑰:" + bPublicKey);
        System.err.println("乙方私鑰:" + bPrivateKey);
        String input = "鄂爾多斯市烏蘭煤炭(集團)有限責任公司溫家樑三號煤礦#C1801003133#蒙AL2928#100.0#2018-12-18 14:52:44#54.82#-100.0#不粘煤#2018-12-18 15:01:40#1.0#李四#6#\"";
        System.out.println("原文: " + input);
// 由甲方公鑰,乙方私鑰構建密文
        String aCode = DMdemo.encrypt(input, aPublicKey, bPrivateKey);
        System.out.println("由甲方公鑰,乙方私鑰構建密文: " + aCode);
        System.out.println(aCode.length());
// 由乙方公鑰,甲方私鑰解密
        String aDecode = DMdemo.decrypt(aCode, bPublicKey, aPrivateKey);
        System.out.println("由乙方公鑰,甲方私鑰解密: " + aDecode); System.err.println(" ===============反過來加密解密================== ");
// 由乙方公鑰,甲方私鑰構建密文
        String bCode = DMdemo.encrypt(input, bPublicKey, aPrivateKey);
        System.out.println("由乙方公鑰,甲方私鑰構建密文: " + bCode);
        System.out.println(bCode.length());
// 由甲方公鑰,乙方私鑰解密
        String bDecode = DMdemo.decrypt(bCode, aPublicKey, bPrivateKey);
        System.out.println("由甲方公鑰,乙方私鑰解密: " + bDecode);
/**
 * 如我代碼證實,甲乙雙方在獲得對方公鑰後可以對發送給對方的數據加密,同時也能對接收到的數據解密,達到了數據安全通信的目的!
 */
    }}

 

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章