0

重複の可能性:
Java で 3des 暗号化/復号化を使用するにはどうすればよいですか?

javaで3DESを使用してテキスト文字列を暗号化/復号化するにはどうすればよいですか?


私は自分の答えを見つけました。これを尋ねたときに表示されなかった重複した質問。

Java で 3des 暗号化/復号化を使用するにはどうすればよいですか?

4

4 に答える 4

2

From an old code:

    public void testSymCypher(SecretKey k, String str)
        throws BadPaddingException, IllegalBlockSizeException,
        InvalidAlgorithmParameterException, InvalidKeyException,
        NoSuchAlgorithmException, NoSuchPaddingException
{
    Cipher cip = Cipher.getInstance("DESede/CBC/PKCS5Padding");
    cip.init(Cipher.ENCRYPT_MODE,k);
    byte[] ciphered = cip.doFinal(str.getBytes());
    byte iv[] = cip.getIV();

    // printing the ciphered string
    printHexadecimal(ciphered);

    IvParameterSpec dps = new IvParameterSpec(iv);
    cip.init(Cipher.DECRYPT_MODE,k,dps);
    byte[] deciphered = cip.doFinal(ciphered);

    // printing the deciphered string
    printHexadecimal(deciphered);
}

Notice than other usage of DESede are available in Java JDK 6:

  • DESede/CBC/NoPadding (168)
  • DESede/CBC/PKCS5Padding (168)

There is also ECB mode available (but be carreful to not use it twice !!), you don't need to use iv part in this case:

  • DESede/ECB/NoPadding (168)
  • DESede/ECB/PKCS5Padding (168)

To generate key for DESede:

    KeyGenerator generatorDes = KeyGenerator.getInstance("DESede");
    SecretKey skaes = generatorDes.generateKey();

Finally I recommand reading this document from SUN if you need to work on Java and Cryptography

于 2009-10-29T18:59:56.977 に答える
1

この小さなヘルパー クラスを、文字列から 16 進文字列へ、およびその逆のパスワード ベースの DES 暗号化に使用します。ただし、これを 3DES で機能させる方法はわかりません。

import java.security.spec.KeySpec;

import javax.crypto.Cipher;
import javax.crypto.SecretKey;
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.PBEKeySpec;
import javax.crypto.spec.PBEParameterSpec;

public class DesHelper {
    private static org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(DesHelper.class);

    static final byte[] SALT = { (byte) 0x09, /* snip - randomly chosen but static salt*/ };
    static final int ITERATIONS = 11;

    private Cipher _ecipher;
    private Cipher _dcipher;

    public DesHelper(final String passphrase) {
        try {
            final PBEParameterSpec params = new PBEParameterSpec(SALT, ITERATIONS);

            final KeySpec keySpec = new PBEKeySpec(passphrase.toCharArray());
            final SecretKey key = SecretKeyFactory.getInstance("PBEWithMD5AndDES")
                    .generateSecret(keySpec);

            _ecipher = Cipher.getInstance(key.getAlgorithm());
            _dcipher = Cipher.getInstance(key.getAlgorithm());
            _ecipher.init(Cipher.ENCRYPT_MODE, key, params);
            _dcipher.init(Cipher.DECRYPT_MODE, key, params);

        } catch (final Exception e) {
            throw new RuntimeException(e);
        }
    }

    public String encrypt(final String string) {
        try {
            // Encode the string into bytes using utf-8
            final byte[] bytes = string.getBytes("UTF-8");

            // Encrypt
            final byte[] enc = _ecipher.doFinal(bytes);

            // Encode bytes to base64 to get a string
            return bytesToHex(enc);
        } catch (final Exception e) {
            throw new RuntimeException(e);
        }
    }

    public String decrypt(final String str) {
        try {
            // Decode base64 to get bytes
            final byte[] dec = hexToBytes(str);

            // Decrypt
            final byte[] utf8 = _dcipher.doFinal(dec);

            // Decode using utf-8
            return new String(utf8, "UTF8");
        } catch (final Exception e) {
            log.info("decrypting string failed: " + str + " (" + e.getMessage() + ")");
            return null;
        }
    }

    private static String bytesToHex(final byte[] bytes) {
        final StringBuilder buf = new StringBuilder(bytes.length * 2);
        for (final byte b : bytes) {
            final String hex = Integer.toHexString(0xff & b);
            if (hex.length() == 1) {
                buf.append("0");
            }
            buf.append(hex);
        }
        return buf.toString();
    }

    private static byte[] hexToBytes(final String hex) {
        final byte[] bytes = new byte[hex.length() / 2];
        for (int i = 0; i < bytes.length; i++) {
            bytes[i] = (byte) Integer.parseInt(hex.substring(i * 2, i * 2 + 2), 16);
        }
        return bytes;
    }
}

このクラスは次のように使用します。

public static void main(final String[] args) {
    final DesHelper h = new DesHelper("blabla");
    System.out.println(h.decrypt(h.encrypt("foobar")));
}
于 2009-10-29T19:21:01.980 に答える
0

私はこれについて時々記事を書きました。説明と図を含む、機能する完成したコードがある私のブログの次のリンクにアクセスしてください。

私のトリプルDES暗号化の記事を見る、ここにコード

うまくいけば、それがお役に立てば幸いです。

于 2009-10-29T21:21:17.620 に答える
0

また、暗号ブロックサイズの倍数になるように文字列をパディングする必要がないように、ストリーム暗号 (たとえば、3DES ブロック暗号化の上に OFB または CTR モード) を使用することを検討することもできます。

于 2009-10-29T23:21:26.113 に答える