Java에서 문자열을 암호화하는 방법


150

내가 필요한 것은 2D 바코드 (PDF-417)로 표시되는 문자열을 암호화하여 누군가가 스캔 할 아이디어를 얻을 때 읽을 수있는 것이 없습니다.

다른 요구 사항 :

  • 복잡해서는 안됩니다
  • RSA, PKI 인프라, 키 쌍 등으로 구성되어서는 안됩니다.

스누핑하는 사람들을 제거하고 데이터를 얻는 데 관심이있는 다른 회사를 쉽게 해독 할 수있을 정도로 간단해야합니다. 그들은 우리를 부르고 표준을 말하거나 암호 해독에 사용할 수있는 간단한 키를 제공합니다.

아마도 이러한 회사는 다른 기술을 사용할 수 있으므로 특정 플랫폼이나 기술과 관련이없는 표준을 고수하는 것이 좋습니다.

당신은 무엇을 제안합니까? 하고 일부 Java 클래스 있습니까 encrypt()decrypt()높은 보안 표준을 달성 많은 합병증없이는?



경고 . 아래의 많은 답변 은 Java에서 모든 종류의 암호화 를 수행 하는 방법을 보여줍니다 . 답변 수 있습니다 좋은 암호화 관행을 반영하지하고 있습니다 잘 검토되지; 복사 / 붙여 넣기 보안과 같은 것은 없습니다 . 답변 최소한 문자열 변환을 고려해야합니다. 2D 바코드가 포함 된 실제 질문은 너무 광범위하므로 고객 별 솔루션이 필요합니다.
Maarten Bodewes

답변:


156

이 페이지는 Google을 통해 표시되는 첫 번째 페이지이며 모든 구현의 보안 취약점으로 인해 위험에 빠질 수 있으므로 원래 게시물에서 7 년 이 지났으므로 다른 사람들의 암호화에 대한 정보를 추가하기 위해이 게시물을 게시하고 있습니다. 컴퓨터 공학 석사 학위 를 소지하고 암호학을 공부하고 배우는 데 많은 시간을 보냈으므로 인터넷을 더 안전한 곳으로 만들기 위해 2 센트를 던졌습니다.

또한 주어진 상황에서 많은 구현이 안전 할 수 있지만 왜 실수로 실수를 저지르는가? 특별한 이유가없는 한 사용 가능한 가장 강력한 도구를 사용하십시오. 전반적으로 나는 도서관을 사용하고 가능한 한 중요한 세부 사항을 피하는 것이 좋습니다.

업데이트 4/5/18 : 권장 라이브러리를 이해하기 쉽고 Jasypt 에서 Google의 새로운 라이브러리 Tink로 변경하기 위해 일부 부분을 다시 작성했습니다 . 기존 설정에서 Jasypt 를 완전히 제거하는 것이 좋습니다 .

머리말

아래에서는 안전한 대칭 암호화의 기본 사항을 간략하게 설명하고 사람들이 표준 Java 라이브러리를 사용하여 자체적으로 암호화를 구현할 때 온라인에서 흔히 볼 수있는 실수를 지적합니다. Google의 새 라이브러리로 실행되는 모든 세부 정보를 건너 뛰 려면 프로젝트로 가져 와서 모든 암호화에 AES-GCM 모드를 사용하면 안전합니다.

이제 자바에서 암호화하는 방법에 대한 핵심적인 내용을 배우려면 :)

블록 암호

먼저 대칭 키 블록 암호를 선택해야합니다. 블록 암호는 의사 난수 생성에 사용되는 컴퓨터 기능 / 프로그램입니다. Pseudo-Randomness는 Quantum Computer 이외의 컴퓨터가 실제 임의성과 그 차이를 구분할 수없는 가짜 임의성입니다. 블록 암호는 암호화의 구성 요소와 비슷하며 다른 모드 나 체계와 함께 사용하면 암호화를 만들 수 있습니다.

이제 오늘 블록 암호 알고리즘을 사용할 수에 대해에 있는지 확인 NEVER , 반복한다 결코 사용하지 DES를 , 난 절대 사용하지 말 것 3DES를 . Snowden의 NSA 릴리스조차도 가능한 한 유사 의사 랜덤에 가까운 지 확인할 수있는 유일한 블록 암호는 AES 256 입니다. AES 128도 있습니다. 차이점은 AES 256은 256 비트 블록에서 작동하지만 AES 128은 128 블록에서 작동한다는 것입니다. 대체로 AES 128은 일부 약점이 발견되었지만 안전한 것으로 간주되지만 256은 얻을 수있는만큼 견고합니다.

재미있는 사실 DES 는 처음 설립되었을 때 NSA에 의해 파괴되어 실제로 몇 년 동안 비밀을 유지했습니다. 일부 사람들은 여전히 3DES 가 안전 하다고 주장하지만 3DES의 약점을 발견하고 분석 한 연구 논문이 상당히 많습니다 .

암호화 모드

암호화는 블록 암호를 가져와 특정 체계를 사용하여 임의성을 키와 결합하여 키를 아는 한 뒤집을 수있는 무언가를 만들 때 만들어집니다. 이것을 암호화 모드라고합니다.

다음은 암호화 모드와 ECB로 알려진 가장 간단한 모드의 예입니다. 현재 상황을 시각적으로 이해할 수 있습니다.

ECB 모드

온라인에서 가장 일반적으로 볼 수있는 암호화 모드는 다음과 같습니다.

ECB CTR, CBC, GCM

나열된 모드 이외의 다른 모드가 있으며 연구원은 기존 문제를 개선하기 위해 항상 새로운 모드를 향해 노력하고 있습니다.

이제 구현과 안전한 것으로 넘어 갑시다. ECB를 사용하지 마십시오 . 유명한 Linux 펭귄이 보여주는 것처럼 반복되는 데이터를 숨기는 데 나쁩니다 .리눅스 펭귄 예제

Java로 구현할 때 다음 코드를 사용하면 ECB 모드가 기본적으로 설정됩니다.

Cipher cipher = Cipher.getInstance("AES");

...이 취약점은 취약점입니다! 불행히도 이것은 자습서 및 예제에서 StackOverflow 및 온라인 전체에서 볼 수 있습니다.

논체와 IV

ECB 모드에서 발견 된 문제에 대한 응답으로 IV라고도하는 nounces가 생성되었습니다. 아이디어는 새로운 임의의 변수를 생성하고 모든 암호화에 첨부하여 동일한 두 메시지를 암호화 할 때 다르게 표시되도록하는 것입니다. 이 뒤에 숨겨진 아름다움은 IV 또는 논 스가 공개 지식이라는 것입니다. 즉, 침입자는 이것에 액세스 할 수 있지만 키가없는 한 해당 지식으로 아무것도 할 수 없습니다.

내가 볼 수있는 일반적인 문제는 사람들이 코드에서 동일한 고정 값과 같이 IV를 정적 값으로 설정한다는 것입니다. 여기에 IV를 반복하는 순간 실제로 암호화의 전체 보안을 손상시키는 함정이 있습니다.

무작위 IV 생성

SecureRandom randomSecureRandom = SecureRandom.getInstance("SHA1PRNG");
byte[] iv = new byte[cipher.getBlockSize()];
randomSecureRandom.nextBytes(iv);
IvParameterSpec ivParams = new IvParameterSpec(iv);

참고 : SHA1이 손상되었지만 SHA256 을이 유스 케이스에 올바르게 구현하는 방법을 찾지 못했습니다. 누군가 가이 문제를 해결하고 업데이트하려면 멋진 것입니다! 또한 SHA1 공격은 대규모 클러스터에서 크랙하기까지 몇 년이 걸릴 수 있으므로 여전히 비 전통적입니다. 자세한 내용은 여기를 확인하십시오.

CTR 구현

CTR 모드에는 패딩이 필요하지 않습니다.

 Cipher cipher = Cipher.getInstance("AES/CTR/NoPadding");

CBC 구현

CBC 모드를 구현하기로 선택한 경우 다음과 같이 PKCS7Padding을 사용하십시오.

 Cipher cipher = Cipher.getInstance("AES/CBC/PKCS7Padding");

CBC 및 CTR 취약점 및 GCM을 사용해야하는 이유

CBC 및 CTR과 같은 일부 다른 모드는 안전하지만 공격자가 암호화 된 데이터를 뒤집어 해독 할 때 값을 변경하는 문제가 발생합니다. 가상 은행 메시지 "Sell 100"을 암호화한다고 가정 해 봅시다. 암호화 된 메시지는이 "eu23ng"처럼 보이고 공격자가 1 비트를 "eu53ng"로 변경하고 갑자기 메시지를 해독하면 "Sell 900"으로 표시됩니다.

이를 피하기 위해 대부분의 인터넷은 GCM을 사용하며 HTTPS를 볼 때마다 GCM을 사용하고있을 것입니다. GCM은 암호화 된 메시지에 해시로 서명하고이 서명을 사용하여 메시지가 변경되지 않았는지 확인합니다.

복잡성으로 인해 GCM을 구현하지 않을 것입니다. 실수로 IV를 반복하면 GCM의 경우 키를 손상시킬 수 있기 때문에 Google의 새로운 라이브러리 Tink를 사용하는 것이 좋습니다 . 이것이 궁극적 인 보안 결함입니다. 새로운 연구원들은 IV 반복 방지 암호화 모드를 향해 노력하고 있습니다. IV를 반복하더라도 키가 위험하지는 않지만 아직 주류가 아닙니다.

이제 GCM을 구현 하려면 멋진 GCM 구현에 대한 링크가 있습니다. 그러나 보안을 보장 할 수 없거나 제대로 구현되었지만 기본을 잃어 버렸습니다. 또한 GCM에서는 패딩이 없습니다.

Cipher cipher = Cipher.getInstance("AES/GCM/NoPadding");

키와 비밀번호

또 다른 중요한 점은 암호화와 관련하여 키와 암호가 동일하지 않다는 것입니다. 암호화 키는 안전한 것으로 간주 되려면 일정량의 엔트로피와 임의성이 있어야합니다. 따라서 키를 생성하려면 적절한 암호화 라이브러리를 사용해야합니다.

그래서 당신은 실제로 두 가지 구현을 할 수 있습니다. 첫 번째는 RandomKey Generation위해이 StackOverflow 스레드에서 찾은 코드를 사용하는 입니다. 이 솔루션은 안전한 난수 생성기를 사용하여 처음부터 사용할 수있는 키를 만듭니다.

덜 안전한 다른 옵션은 암호와 같은 사용자 입력을 사용하는 것입니다. 우리가 논의한 문제는 암호에 엔트로피가 충분하지 않다는 것입니다. 따라서 암호를 취하고 강화하는 알고리즘 인 PBKDF2 를 사용해야 합니다. 여기 내가 좋아 하는 StackOverflow 구현이 있습니다. 그러나 Google Tink 라이브러리에는이 모든 것이 내장되어 있으므로 활용해야합니다.

안드로이드 개발자

여기서 지적해야 할 중요한 점은 안드로이드 코드가 리버스 엔지니어링 가능하며 대부분의 경우 자바 코드도 마찬가지라는 것입니다. 암호를 일반 텍스트로 코드에 저장하면 의미합니다. 해커가 쉽게 검색 할 수 있습니다. 일반적으로 이러한 유형의 암호화에는 비대칭 암호화 등을 사용하려고합니다. 이것은이 게시물의 범위를 벗어나므로 다이빙을 피할 것입니다.

2013 년흥미로운 기사 : 안드로이드에서 구현 한 Crypto 구현의 88 %가 부적절하게 수행되었다고 지적합니다.

마지막 생각들

다시 한 번 암호화를 위해 Java 라이브러리를 직접 구현하지 말고 Google Tink을 사용 하는 것이 좋습니다. 모든 알고리즘을 올바르게 구현하는 데 실제로 많은 노력을 기울 였으므로 두통을 덜어 줄 것입니다. 그런 다음에도 Tink github에서 발생하는 문제, 여기 저기 취약점 팝업을 확인하십시오.

질문이나 의견이 있으시면 언제든지 의견을 보내주십시오! 보안은 항상 변하고 있습니다. 보안을 유지하려면 최선을 다해야합니다. :)


15
이것은 내가 본 것 중 가장 깨끗한 것입니다.
Seraf

1
@SabirKhan 그것은 우려의 원인이 될 수 있지만 핵심 알고리즘은 여전히 ​​깨지지 않았으므로 걱정하지 않아도됩니다. 신뢰할 수없는 경우 github.com/google/keyczar 도 확인하십시오 .Google 보안 팀에서 개발했습니다.
Konstantino Sparakis

1
@ KonstantinoSparakis : Jasypt의 BasicTextEncryptor 및 StrongTextEncryptor에 대한 문서를 잘못 해석하지 않은 경우 해당 클래스 는 암호화에 DES3DES 를 사용합니다. 이는 독자에게 사용하지 말라고 정확하게 지시합니다. IMO 주어진 코드 예제를 Jasypt의 StandardPBEStringEncryptor를 사용하고 수동으로 사용할 AES 알고리즘을 정의하는 코드 예제로 바꿔야합니다.
xpages-noob

1
@ xpages-noob 게시물을 업데이트했습니다. 실제로 최신 암호화 지원 라이브러리 인 Google Tink를 찾았으므로 확인해야합니다!
Konstantino Sparakis

2
AES 블록 크기는 128 비트입니다. AES 256에서 키 크기는 256 비트입니다. 마찬가지로, AES 192 또한 AES 128은 자바 8 이후 getInstanceStrong()의 방법은 CipherSHA1PRNG 이상보다 더 낫다입니다
Saptarshi 바수

110

DES , 3DES 또는 AES 와 같이 널리 사용 가능한 표준 대칭 사이퍼를 사용하는 것이 좋습니다 . 이것이 가장 안전한 알고리즘은 아니지만 많은 구현이 있으며 바코드의 정보를 해독 해야하는 사람에게 열쇠를 주면됩니다. javax.crypto.Cipher 는 여기서 작업하고 싶습니다.

암호화 할 바이트가 있다고 가정합니다

byte[] input;

다음으로 키와 초기화 벡터 바이트 가 필요합니다.

byte[] keyBytes;
byte[] ivBytes;

이제 선택한 알고리즘의 암호를 초기화 할 수 있습니다.

// wrap key data in Key/IV specs to pass to cipher
SecretKeySpec key = new SecretKeySpec(keyBytes, "DES");
IvParameterSpec ivSpec = new IvParameterSpec(ivBytes);
// create the cipher with the algorithm you choose
// see javadoc for Cipher class for more info, e.g.
Cipher cipher = Cipher.getInstance("DES/CBC/PKCS5Padding");

암호화는 다음과 같습니다.

cipher.init(Cipher.ENCRYPT_MODE, key, ivSpec);
byte[] encrypted= new byte[cipher.getOutputSize(input.length)];
int enc_len = cipher.update(input, 0, input.length, encrypted, 0);
enc_len += cipher.doFinal(encrypted, enc_len);

그리고 이와 같은 해독 :

cipher.init(Cipher.DECRYPT_MODE, key, ivSpec);
byte[] decrypted = new byte[cipher.getOutputSize(enc_len)];
int dec_len = cipher.update(encrypted, 0, enc_len, decrypted, 0);
dec_len += cipher.doFinal(decrypted, dec_len);

9
이 예제를 업데이트하여 DESede알고리즘 을 참조하도록 제안 할 수 있습니까 ? 이것은 대중적인 질문과 답변이기 때문에 오늘날의 표준으로 인해 암호가 너무 약하기 때문에 사람들이 DES를 사용하도록 권장하는 것은 부끄러운 일입니다.
Duncan Jones

javax.crypto.BadPaddingException에 문제가있는 경우 : 선언하는 동안 최종 블록이 제대로 채워지지 않음
curiousity

2
@Duncan 실제로 DES가 약하지만 AES는 (TipleDES 일명) DESede 이상 바람직 할 것입니다 가정 : http://security.stackexchange.com/a/26181/69785
Piovezan

2
이것은 AES / GCM / NoPadding을 업데이트해야 DES가 bruteforce 공격에 취약, TRIPLEDES 중 하나를 사용하지 않는 것이 좋습니다
Konstantino Sparakis

1
아래 Konstantino Sparakis의 답변은 이것보다 훨씬 낫습니다.
Steve

22

경고

이것을 일종의 보안 측정으로 사용하지 마십시오.

이 게시물의 암호화 메커니즘은 일회용 패드이므로 공격자가 2 개의 암호화 된 메시지를 사용하여 비밀 키를 쉽게 복구 할 수 있습니다. XOR 2 암호화 된 메시지와 키를 얻습니다. 간단합니다!

무사에 의해 지적


lib의 또 다른 JAR을 피하기 위해 Sun의 JRE에있는 Sun의 Base64Encoder / Decoder를 사용하고 있습니다. OpenJDK 또는 다른 JRE를 사용한다는 점에서 위험합니다. 게다가 Encoder / Decoder와 함께 Apache commons lib를 사용해야하는 또 다른 이유가 있습니까?

public class EncryptUtils {
    public static final String DEFAULT_ENCODING = "UTF-8"; 
    static BASE64Encoder enc = new BASE64Encoder();
    static BASE64Decoder dec = new BASE64Decoder();

    public static String base64encode(String text) {
        try {
            return enc.encode(text.getBytes(DEFAULT_ENCODING));
        } catch (UnsupportedEncodingException e) {
            return null;
        }
    }//base64encode

    public static String base64decode(String text) {
        try {
            return new String(dec.decodeBuffer(text), DEFAULT_ENCODING);
        } catch (IOException e) {
            return null;
        }
    }//base64decode

    public static void main(String[] args) {
        String txt = "some text to be encrypted";
        String key = "key phrase used for XOR-ing";
        System.out.println(txt + " XOR-ed to: " + (txt = xorMessage(txt, key)));

        String encoded = base64encode(txt);       
        System.out.println(" is encoded to: " + encoded + " and that is decoding to: " + (txt = base64decode(encoded)));
        System.out.print("XOR-ing back to original: " + xorMessage(txt, key));
    }

    public static String xorMessage(String message, String key) {
        try {
            if (message == null || key == null) return null;

            char[] keys = key.toCharArray();
            char[] mesg = message.toCharArray();

            int ml = mesg.length;
            int kl = keys.length;
            char[] newmsg = new char[ml];

            for (int i = 0; i < ml; i++) {
                newmsg[i] = (char)(mesg[i] ^ keys[i % kl]);
            }//for i

            return new String(newmsg);
        } catch (Exception e) {
            return null;
        }
    }//xorMessage
}//class

1
또한 sun.misc.BASE64Encoder를 통해이 솔루션 제안을 사용했지만 인코딩하기 위해 다소 큰 문자열을 사용할 때 인코더는 청크 된 문자열 (각 76 문자)을 반환했습니다. 그런 다음 비 청킹 인코딩 방법을 제공하는 Apache Commons Codec Base64로 전환했습니다!
basZero

78
설명 된 암호화 메커니즘은 두 번 이상 사용될 경우 매우 위험합니다. 그것이 일회용 패드라고 불리는 이유입니다. 비밀 키는 2 개의 암호화 된 메시지를 사용하여 공격자가 쉽게 복구 할 수 있습니다. xor 2 암호화 된 메시지와 키를 얻습니다. 간단합니다!
xtrem

3
PDF-417 2D 바코드로 작성된 내용을 읽지 못하게하는 것만으로도 무겁지 않아야합니다. 그리고 어쨌든, 누군가에게 중요하지 않은 몇 가지 색인 만 있습니다.
ante.sabo

2
확인. 누군가가 이것을 암호화 메커니즘으로 사용한다고 우려했습니다.
xtrem

암호화의 경우 인코더 (예 : BASE64Encoder)가 무차별 대입 공격을 피할 수 있습니다.
Jagrut Dalwadi

13

고맙습니다 ive 누군가가 사용자를 찾은 코드를 사용 하여이 클래스를 만들었습니다.

객체 암호

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.security.InvalidAlgorithmParameterException;
import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException;

import javax.crypto.BadPaddingException;
import javax.crypto.Cipher;
import javax.crypto.IllegalBlockSizeException;
import javax.crypto.NoSuchPaddingException;
import javax.crypto.ShortBufferException;
import javax.crypto.spec.DESKeySpec;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;


public class ObjectCrypter {

private Cipher deCipher;
private Cipher enCipher;
private SecretKeySpec key;
private IvParameterSpec ivSpec;


public ObjectCrypter(byte[] keyBytes,   byte[] ivBytes) {
    // wrap key data in Key/IV specs to pass to cipher


     ivSpec = new IvParameterSpec(ivBytes);
    // create the cipher with the algorithm you choose
    // see javadoc for Cipher class for more info, e.g.
    try {
         DESKeySpec dkey = new  DESKeySpec(keyBytes);
          key = new SecretKeySpec(dkey.getKey(), "DES");
         deCipher = Cipher.getInstance("DES/CBC/PKCS5Padding");
         enCipher = Cipher.getInstance("DES/CBC/PKCS5Padding");
    } catch (NoSuchAlgorithmException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (NoSuchPaddingException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (InvalidKeyException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}
public byte[] encrypt(Object obj) throws InvalidKeyException, InvalidAlgorithmParameterException, IOException, IllegalBlockSizeException, ShortBufferException, BadPaddingException {
    byte[] input = convertToByteArray(obj);
    enCipher.init(Cipher.ENCRYPT_MODE, key, ivSpec);

    return enCipher.doFinal(input);




//  cipher.init(Cipher.ENCRYPT_MODE, key, ivSpec);
//  byte[] encypted = new byte[cipher.getOutputSize(input.length)];
//  int enc_len = cipher.update(input, 0, input.length, encypted, 0);
//  enc_len += cipher.doFinal(encypted, enc_len);
//  return encypted;


}
public Object decrypt( byte[]  encrypted) throws InvalidKeyException, InvalidAlgorithmParameterException, IllegalBlockSizeException, BadPaddingException, IOException, ClassNotFoundException {
    deCipher.init(Cipher.DECRYPT_MODE, key, ivSpec);

    return convertFromByteArray(deCipher.doFinal(encrypted));

}



private Object convertFromByteArray(byte[] byteObject) throws IOException,
        ClassNotFoundException {
    ByteArrayInputStream bais;

    ObjectInputStream in;
    bais = new ByteArrayInputStream(byteObject);
    in = new ObjectInputStream(bais);
    Object o = in.readObject();
    in.close();
    return o;

}



private byte[] convertToByteArray(Object complexObject) throws IOException {
    ByteArrayOutputStream baos;

    ObjectOutputStream out;

    baos = new ByteArrayOutputStream();

    out = new ObjectOutputStream(baos);

    out.writeObject(complexObject);

    out.close();

    return baos.toByteArray();

}


}

여기에 관련 질문을 게시했습니다 !
user2023507

암호화 및 암호 해독 중에 다른 키를 전달하면 텍스트가 다시 반환되지 않아야합니까? 그것은 여기서 일어나지 않는 것 같습니다. 추신 :이 테스트를 수행하기 위해이 클래스의 다른 객체를 사용하고 있습니다.
instanceOfObject

6

2019 년 12 월 12 일 업데이트

CBC와 같은 다른 모드와 달리 GCM 모드에서는 IV를 예측할 수 없습니다. 유일한 요구 사항은 IV가 주어진 키를 가진 각 호출에 대해 고유해야한다는 것입니다. 주어진 키에 대해 한 번 반복되면 보안이 손상 될 수 있습니다. 이를 달성하는 쉬운 방법은 아래에 표시된 것처럼 강력한 의사 난수 생성기에서 난수 IV를 사용하는 것입니다.

시퀀스 또는 타임 스탬프를 IV로 사용하는 것도 가능하지만 들리는 것처럼 사소한 것은 아닙니다. 예를 들어, 시스템이 지속적 저장소에서 IV로 이미 사용 된 시퀀스를 올바르게 추적하지 않으면 시스템을 재부팅 한 후 호출로 IV를 반복 할 수 있습니다. 마찬가지로, 완벽한 시계는 없습니다. 컴퓨터 시계 등 조정

또한 2 ^ 32 호출마다 키를 회전해야합니다. IV 요구 사항에 대한 자세한 내용은이 답변NIST 권장 사항을 참조하십시오 .


이것은 다음 사항을 고려하여 방금 Java 8로 작성한 암호화 및 암호 해독 코드입니다. 누군가가 이것을 유용하게 사용하기를 바랍니다.

  1. 암호화 알고리즘 : 256 비트 키를 가진 블록 암호 AES는 충분히 안전한 것으로 간주됩니다. 완전한 메시지를 암호화하려면 모드를 선택해야합니다. 기밀성과 무결성을 모두 제공하는 인증 된 암호화가 권장됩니다. GCM, CCM 및 EAX는 가장 일반적으로 사용되는 인증 된 암호화 모드입니다. GCM이 일반적으로 선호되며 GCM 전용 지침을 제공하는 인텔 아키텍처에서 성능이 우수합니다. 이 세 가지 모드는 모두 CTR 기반 (카운터 기반) 모드이므로 패딩이 필요하지 않습니다. 결과적으로 패딩 관련 공격에 취약하지 않습니다.

  2. GCM에는 초기화 벡터 (IV)가 필요합니다. IV는 비밀이 아닙니다. 유일한 요구 사항은 무작위이거나 예측할 수 없어야합니다. Java SecuredRandom에서이 클래스는 암호화 적으로 강력한 의사 난수를 생성합니다. 의사 난수 생성 알고리즘은 getInstance()메소드 에서 지정할 수 있습니다 . 그러나 Java 8 이후로 권장되는 방법은 getInstanceStrong()구성 및 제공하는 가장 강력한 알고리즘 을 사용하는 방법 을 사용 하는 것입니다.Provider

  3. NIST는 상호 ​​운용성, 효율성 및 단순 설계를 촉진하기 위해 GCM에 96 비트 IV를 권장합니다.

  4. 추가적인 보안을 보장하기 위해, 다음 구현 SecureRandom에서 의사 임의 바이트 생성의 2 ^ 16 바이트마다 생성 된 후에 다시 시드됩니다.

  5. 수신자는 IV를 알아야 암호 텍스트를 해독 할 수 있습니다. 따라서 IV는 암호문과 함께 전송되어야합니다. 일부 구현에서는 IV를 AD (Associated Data)로 전송합니다. 즉, 암호 텍스트와 IV 모두에서 인증 태그가 계산됩니다. 그러나 필수는 아닙니다. 의도적 인 공격이나 네트워크 / 파일 시스템 오류로 인해 전송 중에 IV가 변경되면 인증 태그 유효성 검사가 실패하기 때문에 IV에 암호 텍스트를 미리 붙일 수 있습니다.

  6. 문자열을 변경할 수 없으므로 문자열을 사용하여 일반 텍스트 메시지 나 키를 보관할 수 없습니다. 따라서 사용 후에는 문자열을 지울 수 없습니다. 이러한 정리되지 않은 문자열은 메모리에 남아 힙 덤프에 표시 될 수 있습니다. 같은 이유로, 이러한 암호화 또는 암호 해독 방법을 호출하는 클라이언트는 더 이상 필요하지 않은 메시지 또는 키를 보유한 모든 변수 또는 배열을 지워야합니다.

  7. 일반적인 권장 사항에 따라 코드에 제공자를 하드 코딩하지 않았습니다.

  8. 마지막으로 네트워크 또는 스토리지를 통한 전송을 위해서는 키 또는 암호 텍스트가 Base64 인코딩을 사용하여 인코딩되어야합니다. Base64에 대한 자세한 내용은 여기를 참조하십시오 . Java 8 접근 방식을 따라야합니다

바이트 배열은 다음을 사용하여 지울 수 있습니다.

Arrays.fill(clearTextMessageByteArray, Byte.MIN_VALUE);

그러나, 자바 (8)로, 취소하는 쉬운 방법이 없습니다 SecretKeyspecSecretKey이 두 인터페이스의 구현으로 방법을 구현하지 않는 것 destroy()인터페이스를 Destroyable. 다음 코드에서는 리플렉션 SecretKeySpecSecretKey사용 을 지우는 별도의 메서드가 작성되었습니다 .

아래 언급 된 두 가지 접근 방법 중 하나를 사용하여 키를 생성해야합니다.

키는 비밀번호와 같은 비밀이지만 사람이 사용하는 비밀번호와 달리 키는 암호화 알고리즘에 의해 사용되므로 위의 방법으로 만 생성해야합니다.

package com.sapbasu.javastudy;

import java.lang.reflect.Field;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.Optional;

import javax.crypto.Cipher;
import javax.crypto.spec.GCMParameterSpec;
import javax.crypto.spec.SecretKeySpec;

public class Crypto {

  private static final int AUTH_TAG_SIZE = 128; // bits

  // NIST recommendation: "For IVs, it is recommended that implementations
  // restrict support to the length of 96 bits, to
  // promote interoperability, efficiency, and simplicity of design."
  private static final int IV_LEN = 12; // bytes

  // number of random number bytes generated before re-seeding
  private static final double PRNG_RESEED_INTERVAL = Math.pow(2, 16);

  private static final String ENCRYPT_ALGO = "AES/GCM/NoPadding";

  private static final List<Integer> ALLOWED_KEY_SIZES = Arrays
      .asList(new Integer[] {128, 192, 256}); // bits

  private static SecureRandom prng;

  // Used to keep track of random number bytes generated by PRNG
  // (for the purpose of re-seeding)
  private static int bytesGenerated = 0;

  public byte[] encrypt(byte[] input, SecretKeySpec key) throws Exception {

    Objects.requireNonNull(input, "Input message cannot be null");
    Objects.requireNonNull(key, "key cannot be null");

    if (input.length == 0) {
      throw new IllegalArgumentException("Length of message cannot be 0");
    }

    if (!ALLOWED_KEY_SIZES.contains(key.getEncoded().length * 8)) {
      throw new IllegalArgumentException("Size of key must be 128, 192 or 256");
    }

    Cipher cipher = Cipher.getInstance(ENCRYPT_ALGO);

    byte[] iv = getIV(IV_LEN);

    GCMParameterSpec gcmParamSpec = new GCMParameterSpec(AUTH_TAG_SIZE, iv);

    cipher.init(Cipher.ENCRYPT_MODE, key, gcmParamSpec);
    byte[] messageCipher = cipher.doFinal(input);

    // Prepend the IV with the message cipher
    byte[] cipherText = new byte[messageCipher.length + IV_LEN];
    System.arraycopy(iv, 0, cipherText, 0, IV_LEN);
    System.arraycopy(messageCipher, 0, cipherText, IV_LEN,
        messageCipher.length);
    return cipherText;
  }

  public byte[] decrypt(byte[] input, SecretKeySpec key) throws Exception {
    Objects.requireNonNull(input, "Input message cannot be null");
    Objects.requireNonNull(key, "key cannot be null");

    if (input.length == 0) {
      throw new IllegalArgumentException("Input array cannot be empty");
    }

    byte[] iv = new byte[IV_LEN];
    System.arraycopy(input, 0, iv, 0, IV_LEN);

    byte[] messageCipher = new byte[input.length - IV_LEN];
    System.arraycopy(input, IV_LEN, messageCipher, 0, input.length - IV_LEN);

    GCMParameterSpec gcmParamSpec = new GCMParameterSpec(AUTH_TAG_SIZE, iv);

    Cipher cipher = Cipher.getInstance(ENCRYPT_ALGO);
    cipher.init(Cipher.DECRYPT_MODE, key, gcmParamSpec);

    return cipher.doFinal(messageCipher);
  }

  public byte[] getIV(int bytesNum) {

    if (bytesNum < 1) throw new IllegalArgumentException(
        "Number of bytes must be greater than 0");

    byte[] iv = new byte[bytesNum];

    prng = Optional.ofNullable(prng).orElseGet(() -> {
      try {
        prng = SecureRandom.getInstanceStrong();
      } catch (NoSuchAlgorithmException e) {
        throw new RuntimeException("Wrong algorithm name", e);
      }
      return prng;
    });

    if (bytesGenerated > PRNG_RESEED_INTERVAL || bytesGenerated == 0) {
      prng.setSeed(prng.generateSeed(bytesNum));
      bytesGenerated = 0;
    }

    prng.nextBytes(iv);
    bytesGenerated = bytesGenerated + bytesNum;

    return iv;
  }

  private static void clearSecret(Destroyable key)
      throws IllegalArgumentException, IllegalAccessException,
      NoSuchFieldException, SecurityException {
    Field keyField = key.getClass().getDeclaredField("key");
    keyField.setAccessible(true);
    byte[] encodedKey = (byte[]) keyField.get(key);
    Arrays.fill(encodedKey, Byte.MIN_VALUE);
  }
}

암호화 키는 주로 두 가지 방법으로 생성 할 수 있습니다.

  • 비밀번호없이

    KeyGenerator keyGen = KeyGenerator.getInstance("AES");
    keyGen.init(KEY_LEN, SecureRandom.getInstanceStrong());
    SecretKey secretKey = keyGen.generateKey();
    SecretKeySpec secretKeySpec = new SecretKeySpec(secretKey.getEncoded(),
        "AES");
    Crypto.clearSecret(secretKey);
    // After encryption or decryption with key
    Crypto.clearSecret(secretKeySpec);
  • 비밀번호로

    SecureRandom random = SecureRandom.getInstanceStrong();
    byte[] salt = new byte[32];
    random.nextBytes(salt);
    PBEKeySpec keySpec = new PBEKeySpec(password, salt, iterations, 
       keyLength);
    SecretKeyFactory keyFactory = 
        SecretKeyFactory.getInstance("PBKDF2WithHmacSHA256");
    SecretKey secretKey = keyFactory.generateSecret(keySpec);
    SecretKeySpec secretKeySpec = new SecretKeySpec(secretKey.getEncoded(),
        "AES");
    Crypto.clearSecret(secretKey);
    // After encryption or decryption with key
    Crypto.clearSecret(secretKeySpec);

주석을 기반으로 업데이트

@MaartenBodewes가 지적한 것처럼 내 대답은 String질문에 필요한 것을 처리하지 못했습니다 . 따라서 누군가 가이 답변을 우연히 발견하고 처리에 대해 궁금해하는 경우를 대비하여 그 격차를 메우려 고합니다 String.

답변의 앞부분에서 알 수 있듯이 a에서 민감한 정보를 처리하는 String것은 일반적으로 String불변이므로 사용 후에는 정보를 지울 수 없습니다. 그리고 우리가 알다시피, String강력한 참조가 없더라도 가비지 수집기는 즉시 힙을 제거하기 위해 서두르지 않습니다. 따라서 String프로그램에 액세스 할 수없는 경우에도 메모리에서 알 수없는 시간 동안 계속 유지됩니다. 그 문제는 해당 기간 동안 힙 덤프가 중요한 정보를 공개한다는 것입니다. 따라서 모든 민감한 정보를 바이트 배열 또는 char 배열로 처리 한 다음 목적이 달성되면 배열을 0으로 채우는 것이 항상 좋습니다.

그러나 모든 지식을 가지고 암호화 할 민감한 정보가에있는 상황에서 여전히 끝내면 String먼저 바이트 배열로 변환하고 위에서 소개 한 encryptdecrypt함수를 호출해야합니다 . 다른 입력 키는 위에 제공된 코드 스 니펫을 사용하여 생성 할 수 있습니다.

String다음과 같은 방법으로 A 를 바이트로 변환 할 수 있습니다.

byte[] inputBytes = inputString.getBytes(StandardCharsets.UTF_8);

Java 8 String부터는 UTF-16인코딩 과 함께 내부적으로 힙에 저장됩니다 . 그러나 UTF-8여기서는 UTF-16특히 ASCII 문자의 경우보다 적은 공간을 차지하므로 여기에서 사용 했습니다 .

마찬가지로, 암호화 된 바이트 배열은 다음과 같이 문자열로 변환 될 수도 있습니다.

String encryptedString = new String(encryptedBytes, StandardCharsets.UTF_8);

1
현재 암호 관행을 준수하는 것처럼이 답변을 찬성하고 싶을 때 문자열 처리가 전혀 없으므로 GCM 모드 사용 방법에 대한 설명과 비슷합니다. 따라서 질문대답 하지 못한다 .
Maarten Bodewes

1
@MaartenBodewes 의견을 검토하고 공유하는 데 시간을 내 주셔서 감사합니다. String위에서 만든 함수를 사용하여 암호화하는 것이 쉽지 않을 것이라는 것을 이해하면서 이것을 썼습니다 . 그러나 귀하의 의견을 읽은 후 다시 살펴보면 명확하지 않을 수도 있음을 이해합니다. 이러한 세부 정보를 추가하도록 반드시 편집하겠습니다.
Saptarshi Basu

5

이건 어때요:

private static byte[] xor(final byte[] input, final byte[] secret) {
    final byte[] output = new byte[input.length];
    if (secret.length == 0) {
        throw new IllegalArgumentException("empty security key");
    }
    int spos = 0;
    for (int pos = 0; pos < input.length; ++pos) {
        output[pos] = (byte) (input[pos] ^ secret[spos]);
        ++spos;
        if (spos >= secret.length) {
            spos = 0;
        }
    }
    return output;
}

나를 위해 잘 작동하고 다소 컴팩트합니다.


입력 매개 변수 secret == null 또는 input == null이면 어떻게됩니까? 바이트 대신 문자열로 작업하는 것은 괜찮지 만 제 경우에는 관련이 없었습니다. 중요한 것은 문자 인코딩이 가능한 모든 장치에서 읽고 읽을 수 있어야한다는 것입니다.
ante.sabo

@ ante.sabo는 분명히 NPE를 던질 것입니다. 이것은 NULL과 관련된 유일한 것입니다.
Miha_x64

길이로는 input.length <= secret.length보유하지 않고는 secret이 보안과라고 다시 사용할 있나이다 one-time-pad. 이 경우 Vigenère 암호input.length > secret.length 의 변형이며 매우 약한 것으로 간주됩니다.
trichner

5

Jasypt 를 사용할 수 있습니다

Jasypt를 사용하면 암호를 암호화하고 확인하는 것이 간단 할 수 있습니다.

StrongTextEncryptor textEncryptor = new StrongTextEncryptor();
textEncryptor.setPassword(myEncryptionPassword);

암호화 :

String myEncryptedText = textEncryptor.encrypt(myText);

복호화 :

String plainText = textEncryptor.decrypt(myEncryptedText);

그레들 :

compile group: 'org.jasypt', name: 'jasypt', version: '1.9.2'

풍모:

Jasypt는 간편한 단방향 (다이제스트) 및 양방향 암호화 기술을 제공합니다.

기본 Java VM뿐만 아니라 모든 JCE 공급자와 함께 사용할 수있는 개방형 API입니다. Jasypt는 Bouncy Castle과 같은 잘 알려진 공급자와 쉽게 사용할 수 있습니다. 더 알아보기.

사용자 비밀번호 보안 강화 더 알아보기.

이진 암호화 지원. Jasypt는 바이너리 (바이트 배열)의 다이제스트 및 암호화를 허용합니다. 필요할 때 개체 나 파일을 암호화합니다 (예 : 인터넷을 통해 전송).

숫자 암호화 지원. 텍스트 및 이진 외에도 숫자 값의 다이제스트 및 암호화를 허용합니다 (Hibernate 지속성을 암호화 할 때 BigInteger 및 BigDecimal, 다른 숫자 유형이 지원됨). 더 알아보기.

완전히 스레드 안전합니다.

멀티 프로세서 / 멀티 코어 시스템에서 고성능을 달성하기 위해 암호화 / 디 버스터 풀링을 지원합니다.

경량 ( "라이트") 버전의 라이브러리를 포함하여 모바일 플랫폼과 같은 크기 제한 환경에서 더 나은 관리 효율성을 제공합니다.

암호화를 처음 사용하는 사용자에게는 구성이 필요없는 쉬운 암호화 도구와 고급 사용자를위한 고도로 구성 가능한 표준 암호화 도구를 모두 제공합니다.

암호화 된 방식으로 매핑 된 엔터티의 필드를 유지하기위한 Hibernate 3 및 4 선택적 통합. 필드의 암호화는 최대 절전 모드 매핑 파일에 정의되어 있으며 나머지 응용 프로그램에서는 투명하게 유지됩니다 (민감한 개인 데이터, 읽기 가능한 많은 사용자가있는 데이터베이스에 유용). 텍스트, 이진, 숫자, 부울, 날짜 암호화 ... 자세히 알아보기

Spring 2, Spring 3.0 및 Spring 3.1에 대한 특정 통합 기능을 통해 Spring 애플리케이션에 완벽하게 통합 할 수 있습니다. jasypt의 모든 다이제스트 및 암호화 기는 Spring에서 쉽게 사용할 수 있도록 설계되었습니다 (인스턴스가 주입되고 ...). 또한 스레드로부터 안전하기 때문에 Spring과 같은 싱글 톤 지향 환경에서 동기화 걱정없이 사용할 수 있습니다. 자세히 알아보기 : Spring 2, Spring 3.0, Spring 3.1.

스프링 암호화 (이전의 Acegi Security)는 보안 암호화를위한 암호 암호화 및 일치 작업을 수행하고 안전한 암호 암호화 메커니즘을 사용하여 사용자 암호의 보안을 향상시키고보다 높은 수준의 구성 및 제어를 제공하기위한 선택적 통합입니다. 더 알아보기.

데이터베이스 비밀번호와 같은 민감한 정보를 포함하여 애플리케이션 구성 파일의 일부 또는 전부를 암호화하는 고급 기능을 제공합니다. 암호화 된 구성을 일반 스프링 기반 및 / 또는 최대 절전 모드 지원 응용 프로그램에 완벽하게 통합합니다. 더 알아보기.

사용하기 쉬운 CLI (명령 줄 인터페이스) 도구를 제공하여 개발자가 암호화 된 데이터를 초기화하고 유지 관리 작업 또는 스크립트에 암호화 / 복호화 / 다이제스트 작업을 포함시킬 수 있습니다. 더 알아보기.

보안 애플리케이션에서 URL을보다 강력하게 암호화하기 위해 Apache Wicket에 통합됩니다.

개발자가 자신의 데이터에 대한 실제 작업을 더 잘 이해할 수 있도록 포괄적 인 안내서 및 javadoc 문서.

강력한 문자 집합 지원. 원본 문자 집합에 관계없이 텍스트를 적절히 암호화하고 요약합니다. 인코딩이나 플랫폼 문제없이 일본어, 한국어, 아랍어 등의 언어를 완벽하게 지원합니다.

매우 높은 수준의 구성 기능 : 개발자는 "암호화 기"에 지시하는 것과 같은 트릭을 구현하여 암호화에 사용될 암호를 원격 HTTPS 서버와 같이 요청할 수 있습니다. 보안 요구를 충족시킬 수 있습니다.


1
그러나 어떤 보안이 Jasypt제공됩니까? 나는 그들의 웹 사이트에서 그것을 알아낼 수 없습니다. 선택한 일반 텍스트 공격에서 구별 할 수 없습니까? 청렴? 기밀 유지?
trichner

4

다음은 meta64.com에서 Spring Singleton으로 구현 한 것입니다. 작동하는 각 호출에 대해 사이퍼 인스턴스를 작성하려는 경우 '동기화 된'호출을 제거 할 수 있지만 '암호'는 스레드로부터 안전하지 않습니다.

import java.security.Key;

import javax.crypto.Cipher;
import javax.crypto.spec.SecretKeySpec;
import javax.xml.bind.DatatypeConverter;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;

@Component
@Scope("singleton")
public class Encryptor {

    @Value("${aeskey}")
    private String keyStr;

    private Key aesKey = null;
    private Cipher cipher = null;

    synchronized private void init() throws Exception {
        if (keyStr == null || keyStr.length() != 16) {
            throw new Exception("bad aes key configured");
        }
        if (aesKey == null) {
            aesKey = new SecretKeySpec(keyStr.getBytes(), "AES");
            cipher = Cipher.getInstance("AES");
        }
    }

    synchronized public String encrypt(String text) throws Exception {
        init();
        cipher.init(Cipher.ENCRYPT_MODE, aesKey);
        return toHexString(cipher.doFinal(text.getBytes()));
    }

    synchronized public String decrypt(String text) throws Exception {
        init();
        cipher.init(Cipher.DECRYPT_MODE, aesKey);
        return new String(cipher.doFinal(toByteArray(text)));
    }

    public static String toHexString(byte[] array) {
        return DatatypeConverter.printHexBinary(array);
    }

    public static byte[] toByteArray(String s) {
        return DatatypeConverter.parseHexBinary(s);
    }

    /*
     * DO NOT DELETE
     * 
     * Use this commented code if you don't like using DatatypeConverter dependency
     */
    // public static String toHexStringOld(byte[] bytes) {
    // StringBuilder sb = new StringBuilder();
    // for (byte b : bytes) {
    // sb.append(String.format("%02X", b));
    // }
    // return sb.toString();
    // }
    //
    // public static byte[] toByteArrayOld(String s) {
    // int len = s.length();
    // byte[] data = new byte[len / 2];
    // for (int i = 0; i < len; i += 2) {
    // data[i / 2] = (byte) ((Character.digit(s.charAt(i), 16) << 4) + Character.digit(s.charAt(i +
    // 1), 16));
    // }
    // return data;
    // }
}

3
이것은 끔찍한 ECB 모드로 암호화합니다. 당신은 적어도 CBC 모드 또는 GCM 모드에서 설정되어야한다
Konstantino Sparakis

Konstantinto 제안에 감사드립니다. 나는 그것을 구글 검색하고 "AES"대신 "AES / CBC / PKCS5Padding"을 암호의 초기화 문자열로 사용하는 코드를 찾았지만 더 자세히 살펴볼 것입니다. 또는 원하는 경우 실제 수정 사항을 제공하여 다른 사람이 더 나은 방법을 볼 수 있습니다. 그러나 CBC 세부 사항을 제외하고는 내 솔루션이 가장 간단하고 안전하며 다른 모든 것보다 우선시해야한다고 생각합니다.

그러나 걱정할 필요는 없습니다. Crypto는 복잡한 주제입니다. 안타깝게도이 페이지의 모든 구현은 깨졌으며 슬프게도 Google을 사용하여 "자바 암호화를 수행하는 방법"을 검색 할 때 나타나는 첫 번째 페이지입니다. 기회가 생기면 모든 문제를 해결하려고 노력할 것입니다.
Konstantino Sparakis

내 예는 다음과 같습니다. docs.oracle.com/javase/8/docs/technotes/guides/security/crypto/… 필요한 경우 제외 Cipher.getInstance ( "AES / ECB / PKCS5Padding"); 내 코드는 완벽하게 16 바이트 길이의 암호화 키가있는 일부 속성 파일이 있다고 가정하지만 '사용자 제공'암호에서 문자열을 암호화하기 위해 오라클 페이지 (위 링크)는 그 방법도 보여줍니다.

1
따라서 ECB의 문제점은 주파수 분석에 매우 취약하다는 것입니다. Linux 펭귄의 유명한 예인 blog.filippo.io/the-ecb-penguin 은 이미지가 암호화되어 있지만 펭귄 임을 알 수 있습니다. 내가 나서서 아래 : 제목 아래에 내 생각을 쓴 stackoverflow.com/a/43779197/2607972
Konstantino Sparakis에게

4

여기에 단지와 간단한 솔루션 java.*javax.crypto.*제공 바이트의 암호화를위한 종속 기밀성무결성을 . 킬로바이트 단위의 단문 메시지에 대해 선택된 일반 텍스트 공격 하에서 구별 할 수 없습니다 .

패딩이없는 모드 AES에서 사용 되며 GCM128 비트 키는 PBKDF2많은 반복과 제공된 암호의 정적 소금으로 파생됩니다 . 이렇게하면 무차별 암호를 강제로 입력하고 전체 키에 엔트로피를 분산시킬 수 있습니다.

무작위 초기화 벡터 (IV)가 생성되어 암호문 앞에 붙습니다. 또한 정적 바이트 0x01는 '버전'으로 첫 번째 바이트로 추가됩니다.

전체 메시지는에서 생성 한 메시지 인증 코드 (MAC)로 들어갑니다 AES/GCM.

기밀성무결성을 제공하는 외부 종속성 암호화 클래스가 없습니다 .

package ch.n1b.tcrypt.utils;

import java.nio.charset.StandardCharsets;
import java.security.InvalidAlgorithmParameterException;
import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException;
import java.security.NoSuchProviderException;
import java.security.SecureRandom;
import java.security.spec.InvalidKeySpecException;
import java.security.spec.KeySpec;

import javax.crypto.*;
import javax.crypto.spec.GCMParameterSpec;
import javax.crypto.spec.PBEKeySpec;
import javax.crypto.spec.SecretKeySpec;

/**
 * This class implements AES-GCM symmetric key encryption with a PBKDF2 derived password.
 * It provides confidentiality and integrity of the plaintext.
 *
 * @author Thomas Richner
 * @created 2018-12-07
 */
public class AesGcmCryptor {

    // /crypto/26783/ciphertext-and-tag-size-and-iv-transmission-with-aes-in-gcm-mode
    private static final byte VERSION_BYTE = 0x01;
    private static final int VERSION_BYTE_LENGTH = 1;
    private static final int AES_KEY_BITS_LENGTH = 128;


    // fixed AES-GCM constants
    private static final String GCM_CRYPTO_NAME = "AES/GCM/NoPadding";
    private static final int GCM_IV_BYTES_LENGTH = 12;
    private static final int GCM_TAG_BYTES_LENGTH = 16;

    // can be tweaked, more iterations = more compute intensive to brute-force password
    private static final int PBKDF2_ITERATIONS = 1024;

    // protects against rainbow tables
    private static final byte[] PBKDF2_SALT = hexStringToByteArray("4d3fe0d71d2abd2828e7a3196ea450d4");

    public String encryptString(char[] password, String plaintext) throws CryptoException {

        byte[] encrypted = null;
        try {
            encrypted = encrypt(password, plaintext.getBytes(StandardCharsets.UTF_8));
        } catch (NoSuchAlgorithmException | NoSuchPaddingException | InvalidKeyException //
                | InvalidAlgorithmParameterException | IllegalBlockSizeException | BadPaddingException //
                | InvalidKeySpecException e) {
            throw new CryptoException(e);
        }
        return byteArrayToHexString(encrypted);
    }

    public String decryptString(char[] password, String ciphertext)
            throws CryptoException {

        byte[] ct = hexStringToByteArray(ciphertext);
        byte[] plaintext = null;
        try {
            plaintext = decrypt(password, ct);
        } catch (AEADBadTagException e) {
            throw new CryptoException(e);
        } catch ( //
                NoSuchPaddingException | NoSuchAlgorithmException | InvalidKeySpecException //
                        | InvalidKeyException | InvalidAlgorithmParameterException | IllegalBlockSizeException //
                        | BadPaddingException e) {
            throw new CryptoException(e);
        }
        return new String(plaintext, StandardCharsets.UTF_8);
    }

    /**
     * Decrypts an AES-GCM encrypted ciphertext and is
     * the reverse operation of {@link AesGcmCryptor#encrypt(char[], byte[])}
     *
     * @param password   passphrase for decryption
     * @param ciphertext encrypted bytes
     * @return plaintext bytes
     * @throws NoSuchPaddingException
     * @throws NoSuchAlgorithmException
     * @throws NoSuchProviderException
     * @throws InvalidKeySpecException
     * @throws InvalidAlgorithmParameterException
     * @throws InvalidKeyException
     * @throws BadPaddingException
     * @throws IllegalBlockSizeException
     * @throws IllegalArgumentException           if the length or format of the ciphertext is bad
     * @throws CryptoException
     */
    public byte[] decrypt(char[] password, byte[] ciphertext)
            throws NoSuchPaddingException, NoSuchAlgorithmException, InvalidKeySpecException,
            InvalidAlgorithmParameterException, InvalidKeyException, BadPaddingException, IllegalBlockSizeException {

        // input validation
        if (ciphertext == null) {
            throw new IllegalArgumentException("ciphertext cannot be null");
        }

        if (ciphertext.length <= VERSION_BYTE_LENGTH + GCM_IV_BYTES_LENGTH + GCM_TAG_BYTES_LENGTH) {
            throw new IllegalArgumentException("ciphertext too short");
        }

        // the version must match, we don't decrypt other versions
        if (ciphertext[0] != VERSION_BYTE) {
            throw new IllegalArgumentException("wrong version: " + ciphertext[0]);
        }

        // input seems legit, lets decrypt and check integrity

        // derive key from password
        SecretKey key = deriveAesKey(password, PBKDF2_SALT, AES_KEY_BITS_LENGTH);

        // init cipher
        Cipher cipher = Cipher.getInstance(GCM_CRYPTO_NAME);
        GCMParameterSpec params = new GCMParameterSpec(GCM_TAG_BYTES_LENGTH * 8,
                ciphertext,
                VERSION_BYTE_LENGTH,
                GCM_IV_BYTES_LENGTH
        );
        cipher.init(Cipher.DECRYPT_MODE, key, params);

        final int ciphertextOffset = VERSION_BYTE_LENGTH + GCM_IV_BYTES_LENGTH;

        // add version and IV to MAC
        cipher.updateAAD(ciphertext, 0, ciphertextOffset);

        // decipher and check MAC
        return cipher.doFinal(ciphertext, ciphertextOffset, ciphertext.length - ciphertextOffset);
    }

    /**
     * Encrypts a plaintext with a password.
     * <p>
     * The encryption provides the following security properties:
     * Confidentiality + Integrity
     * <p>
     * This is achieved my using the AES-GCM AEAD blockmode with a randomized IV.
     * <p>
     * The tag is calculated over the version byte, the IV as well as the ciphertext.
     * <p>
     * Finally the encrypted bytes have the following structure:
     * <pre>
     *          +-------------------------------------------------------------------+
     *          |         |               |                             |           |
     *          | version | IV bytes      | ciphertext bytes            |    tag    |
     *          |         |               |                             |           |
     *          +-------------------------------------------------------------------+
     * Length:     1B        12B            len(plaintext) bytes            16B
     * </pre>
     * Note: There is no padding required for AES-GCM, but this also implies that
     * the exact plaintext length is revealed.
     *
     * @param password  password to use for encryption
     * @param plaintext plaintext to encrypt
     * @throws NoSuchAlgorithmException
     * @throws NoSuchProviderException
     * @throws NoSuchPaddingException
     * @throws InvalidAlgorithmParameterException
     * @throws InvalidKeyException
     * @throws BadPaddingException
     * @throws IllegalBlockSizeException
     * @throws InvalidKeySpecException
     */
    public byte[] encrypt(char[] password, byte[] plaintext)
            throws NoSuchAlgorithmException, NoSuchPaddingException,
            InvalidAlgorithmParameterException, InvalidKeyException, BadPaddingException, IllegalBlockSizeException,
            InvalidKeySpecException {

        // initialise random and generate IV (initialisation vector)
        SecretKey key = deriveAesKey(password, PBKDF2_SALT, AES_KEY_BITS_LENGTH);
        final byte[] iv = new byte[GCM_IV_BYTES_LENGTH];
        SecureRandom random = SecureRandom.getInstanceStrong();
        random.nextBytes(iv);

        // encrypt
        Cipher cipher = Cipher.getInstance(GCM_CRYPTO_NAME);
        GCMParameterSpec spec = new GCMParameterSpec(GCM_TAG_BYTES_LENGTH * 8, iv);
        cipher.init(Cipher.ENCRYPT_MODE, key, spec);

        // add IV to MAC
        final byte[] versionBytes = new byte[]{VERSION_BYTE};
        cipher.updateAAD(versionBytes);
        cipher.updateAAD(iv);

        // encrypt and MAC plaintext
        byte[] ciphertext = cipher.doFinal(plaintext);

        // prepend VERSION and IV to ciphertext
        byte[] encrypted = new byte[1 + GCM_IV_BYTES_LENGTH + ciphertext.length];
        int pos = 0;
        System.arraycopy(versionBytes, 0, encrypted, 0, VERSION_BYTE_LENGTH);
        pos += VERSION_BYTE_LENGTH;
        System.arraycopy(iv, 0, encrypted, pos, iv.length);
        pos += iv.length;
        System.arraycopy(ciphertext, 0, encrypted, pos, ciphertext.length);

        return encrypted;
    }

    /**
     * We derive a fixed length AES key with uniform entropy from a provided
     * passphrase. This is done with PBKDF2/HMAC256 with a fixed count
     * of iterations and a provided salt.
     *
     * @param password passphrase to derive key from
     * @param salt     salt for PBKDF2 if possible use a per-key salt, alternatively
     *                 a random constant salt is better than no salt.
     * @param keyLen   number of key bits to output
     * @return a SecretKey for AES derived from a passphrase
     * @throws NoSuchAlgorithmException
     * @throws InvalidKeySpecException
     */
    private SecretKey deriveAesKey(char[] password, byte[] salt, int keyLen)
            throws NoSuchAlgorithmException, InvalidKeySpecException {

        if (password == null || salt == null || keyLen <= 0) {
            throw new IllegalArgumentException();
        }
        SecretKeyFactory factory = SecretKeyFactory.getInstance("PBKDF2WithHmacSHA256");
        KeySpec spec = new PBEKeySpec(password, salt, PBKDF2_ITERATIONS, keyLen);
        SecretKey pbeKey = factory.generateSecret(spec);

        return new SecretKeySpec(pbeKey.getEncoded(), "AES");
    }

    /**
     * Helper to convert hex strings to bytes.
     * <p>
     * May be used to read bytes from constants.
     */
    private static byte[] hexStringToByteArray(String s) {

        if (s == null) {
            throw new IllegalArgumentException("Provided `null` string.");
        }

        int len = s.length();
        if (len % 2 != 0) {
            throw new IllegalArgumentException("Invalid length: " + len);
        }

        byte[] data = new byte[len / 2];
        for (int i = 0; i < len - 1; i += 2) {
            byte b = (byte) toHexDigit(s, i);
            b <<= 4;
            b |= toHexDigit(s, i + 1);
            data[i / 2] = b;
        }
        return data;
    }

    private static int toHexDigit(String s, int pos) {
        int d = Character.digit(s.charAt(pos), 16);
        if (d < 0) {
            throw new IllegalArgumentException("Cannot parse hex digit: " + s + " at " + pos);
        }
        return d;
    }

    private static String byteArrayToHexString(byte[] bytes) {
        StringBuilder sb = new StringBuilder();
        for (byte b : bytes) {
            sb.append(String.format("%02X", b));
        }
        return sb.toString();
    }

    public class CryptoException extends Exception {

        public CryptoException(Throwable cause) {
            super(cause);
        }
    }
}

멋진 CLI가있는 전체 프로젝트는 다음과 같습니다. https://github.com/trichner/tcrypt

편집 : 이제 적절 encryptString하고decryptString


이것은 믿기 힘든 일이야. 감사합니다! 나는 당신의 코드에서 많은 것을 배웠고 BadVersionException Exception 클래스를 만든 후에는 코드가 처음으로 완벽하게 작동했습니다. 우수한!!
Morkus

나는이 시도를 좋아한다. 그 말은 ... 소금은 정적이 아닌 무작위이어야합니다. 반복은 정적 일 수 없습니다. GCM은 이미 태그 계산에 IV를 포함합니다. 버전 번호는 포함되어 있지 않습니다. 이식성을 위해 제공자를 지정하지 않아야합니다. "SunJCE"제공자는이를 지원하는 플랫폼에서 기본값이됩니다. 이 코드에는이 특정 질문에 필요한 메시지 문자열 처리가 포함되어 있지 않습니다 .
Maarten Bodewes

좋아, 내가 더 조금 그것을 청소하고 요청 된 추가 encryptStringdecryptString:)
trichner

이것은 매우 잘 작동했습니다. 코드를 입력하십시오. 이 코드가 제대로 작동하려면 API 19 (Kit Kat) 이상이 필요합니다.
PGMacDesign 19

3

나는 https://www.bouncycastle.org/ 와 같은 것을 사용하는 것을 고려할 것입니다. 그것은 당신이 스누핑으로부터 보호하고 싶어한다는 것을 이해하지만 실제로는 Base64를 사용해도 실제로 정보를 보호 할 수는 없습니다.


1
암호가있는 임의의 암호화 라이브러리를 추천하는 것만으로는 그 질문에 대한 답이 아닙니다. 그 외에도 내장 암호를 사용하지 않는 이유는 무엇입니까?
Maarten Bodewes

2

다음은 Java가 지원하는 내용을 읽을 수있는 링크입니다.

데이터 스트림의 암호화 / 복호화

이 예는 AES, Blowfish, RC2, 3DES 등과 같은 대칭 암호화 알고리즘을 사용하여 많은 양의 데이터를 암호화하는 방법을 보여줍니다. 데이터는 EncryptBytes, EncryptString, EncryptBytesENC 또는 EncryptStringENC와 같은 암호화 방법 중 하나로 청크로 전달됩니다. (메소드 이름은 입력 유형 (문자열 또는 바이트 배열)과 반환 유형 (인코딩 된 문자열 또는 바이트 배열)을 나타냅니다. FirstChunk 및 LastChunk 속성은 청크가 스트림에서 첫 번째, 중간 또는 마지막인지 여부를 나타내는 데 사용됩니다. 기본적으로 FirstChunk와 LastChunk는 모두 true와 같습니다. 즉, 전달 된 데이터가 전체 양임을 의미합니다.

JCERefGuide

자바 암호화 예제


예, Java가 지원하는 암호화가 있습니다. 스트림의 암호화도 요구되지 않았습니다.
Maarten Bodewes

2

많은 사람들이 이미 말했듯이 DES 또는 AES와 같이 과도하게 사용되는 표준 사이퍼를 사용해야합니다.

당신은 암호화 및 사용하여 자바에서 문자열을 해독 할 수있는 방법의 간단한 예를 AES를 .

import org.apache.commons.codec.binary.Base64;

import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;

public class EncryptorDemo {

    public static String encrypt(String key, String randomVector, String value) {
        try {
            IvParameterSpec iv = new IvParameterSpec(randomVector.getBytes("UTF-8"));
            SecretKeySpec skeySpec = new SecretKeySpec(key.getBytes("UTF-8"), "AES");
            Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5PADDING");
            cipher.init(Cipher.ENCRYPT_MODE, skeySpec, iv);
            byte[] encrypted = cipher.doFinal(value.getBytes());
            System.out.println("encrypted text: "  + Base64.encodeBase64String(encrypted));
            return Base64.encodeBase64String(encrypted);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

    public static String decrypt(String key, String randomVector, String encrypted) {
        try {
            IvParameterSpec iv = new IvParameterSpec(randomVector.getBytes("UTF-8"));
            SecretKeySpec skeySpec = new SecretKeySpec(key.getBytes("UTF-8"), "AES");
            Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5PADDING");
            cipher.init(Cipher.DECRYPT_MODE, skeySpec, iv);
            byte[] originalText = cipher.doFinal(Base64.decodeBase64(encrypted));
            System.out.println("decrypted text: "  + new String(originalText));
            return new String(originalText);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

    public static void main(String[] args) {
        String key = "JavasEncryptDemo"; // 128 bit key
        String randomVector = "RandomJavaVector"; // 16 bytes IV
        decrypt(key, randomVector, encrypt(key, randomVector, "Anything you want to encrypt!"));

    }
}

CBC는 더 이상 보안 모드가 아닙니다. 패딩은 Oracle 공격 패딩에 취약합니다. 또한 문자열에서 키와 메시지를 처리하는 것은 안전하지 않습니다. 그들은 문자열 풀에 남아 힙 덤프에 나타납니다
Saptarshi Basu

2
의견을 부탁드립니다. 이것은 사용자가 요청한대로 Java의 암호화 및 암호 해독 방법의 간단한 예입니다. 이 질문은 약 9 년 전에 요청되었으며 그에 근거하여 답변되었습니다. 감사.
viveknaskar

2
예, 이것은 암호화 / 암호 해독을 도입하는 간단한 방법 인 것 같습니다. 나를 위해 매력처럼 일했다 .... 감사합니다.
Codewrapper

0

복사 / 붙여 넣기 솔루션은 다음과 같습니다. 또한 코드를 제공하지 않더라도 @ Konstantino의 답변 을 읽고 투표하는 것이 좋습니다 . 초기화 벡터 (IV)는 소금과 같습니다. 비밀로 유지할 필요는 없습니다. 나는 GCM을 처음 사용하며 AAD는 선택 사항이며 특정 상황에서만 사용됩니다. 환경 변수에 키를 설정하십시오 SECRET_KEY_BASE. KeePass 와 같은 것을 사용 하여 32 문자 암호를 생성하십시오. 이 솔루션은 내 Ruby 솔루션을 모델로합니다.

    public static String encrypt(String s) {
        try {
            byte[] input = s.getBytes("UTF-8");
            String keyString = System.getProperty("SECRET_KEY_BASE", System.getenv("SECRET_KEY_BASE"));
            if (keyString == null || keyString.length() == 0) {
                Logger.error(Utils.class, "encrypt()", "$SECRET_KEY_BASE is not set.");
                return null;
            }
            byte[] keyBytes = keyString.getBytes("UTF-8");
            SecretKeySpec keySpec = new SecretKeySpec(keyBytes, "AES");
            // generate IV
            SecureRandom secureRandom = SecureRandom.getInstanceStrong();
            Cipher cipher = Cipher.getInstance("AES/GCM/NoPadding");
            byte[] ivBytes = new byte[cipher.getBlockSize()];
            secureRandom.nextBytes(ivBytes);
            GCMParameterSpec gcmSpec = new GCMParameterSpec(96, ivBytes); // 96 bit tag length
            cipher.init(Cipher.ENCRYPT_MODE, keySpec, gcmSpec);
            // generate AAD
//          byte[] aadBytes = new byte[cipher.getBlockSize()];
//          secureRandom.nextBytes(aadBytes);
//          cipher.updateAAD(aadBytes);
            // encrypt
            byte[] encrypted = cipher.doFinal(input);
            byte[] returnBytes = new byte[ivBytes.length + encrypted.length];
//          byte[] returnBytes = new byte[ivBytes.length + aadBytes.length + encrypted.length];
            System.arraycopy(ivBytes, 0, returnBytes, 0, ivBytes.length);
//          System.arraycopy(aadBytes, 0, returnBytes, ivBytes.length, aadBytes.length);
            System.arraycopy(encrypted, 0, returnBytes, ivBytes.length, encrypted.length);
//          System.arraycopy(encrypted, 0, returnBytes, ivBytes.length+aadBytes.length, encrypted.length);
            String encryptedString = Base64.getEncoder().encodeToString(returnBytes);
            return encryptedString;
        } catch (UnsupportedEncodingException | NoSuchAlgorithmException | NoSuchPaddingException | InvalidKeyException | 
                InvalidAlgorithmParameterException | IllegalBlockSizeException | BadPaddingException e) {
            Logger.error(Utils.class, "encrypt()", "Could not encrypt string: " + e.getMessage());
            return null;
        }
    }

    public static String decrypt(String s) {
        if (s == null || s.length() == 0) return "";
        try {
            byte[] encrypted = Base64.getDecoder().decode(s);
            String keyString = System.getProperty("SECRET_KEY_BASE", System.getenv("SECRET_KEY_BASE"));
            if (keyString == null || keyString.length() == 0) {
                Logger.error(Utils.class, "encrypt()", "$SECRET_KEY_BASE is not set.");
                return null;
            }
            byte[] keyBytes = keyString.getBytes("UTF-8");
            SecretKeySpec keySpec = new SecretKeySpec(keyBytes, "AES");
            Cipher cipher = Cipher.getInstance("AES/GCM/NoPadding");
            byte[] ivBytes = new byte[cipher.getBlockSize()];
            System.arraycopy(encrypted, 0, ivBytes, 0, ivBytes.length);
            GCMParameterSpec gcmSpec = new GCMParameterSpec(96, ivBytes);
            cipher.init(Cipher.DECRYPT_MODE, keySpec, gcmSpec);
//          cipher.updateAAD(encrypted, ivBytes.length, cipher.getBlockSize());
            byte[] decrypted = cipher.doFinal(encrypted, cipher.getBlockSize(), encrypted.length - cipher.getBlockSize());
//          byte[] decrypted = cipher.doFinal(encrypted, cipher.getBlockSize()*2, encrypted.length - cipher.getBlockSize()*2);
            String decryptedString = new String(decrypted, "UTF-8");
            return decryptedString;
        } catch (NoSuchAlgorithmException | NoSuchPaddingException | UnsupportedEncodingException | InvalidKeyException | 
                InvalidAlgorithmParameterException | IllegalBlockSizeException | BadPaddingException e) {
            Logger.error(Utils.class, "decrypt()", "Could not decrypt string: " + e.getMessage());
            return null;
        }
    }

예를 들면 다음과 같습니다.

    String s = "This is a test.";
    String enc = Utils.encrypt(s);
    System.out.println(enc);
    // fQHfYjbD+xAuN5XzH2ojk/EWNeKXUrKRSfx8LU+5dpuKkM/pueCMBjKCZw==
    String dec = Utils.decrypt(enc);
    System.out.println(dec);
    // This is a test.

-4

예를 들어 암호화 / 암호 해독 코드 생성을 수행하는 자동화 된 도구를 고려할 수 있습니다. https://www.stringencrypt.com/java-encryption/

문자열 또는 파일 암호화에 대해 매번 다른 암호화 및 암호 해독 코드를 생성 할 수 있습니다.

RSA, AES 등을 사용하지 않고 빠른 문자열 암호화와 관련하여 매우 편리합니다.

샘플 결과 :

// encrypted with https://www.stringencrypt.com (v1.1.0) [Java]
// szTest = "Encryption in Java!"
String szTest = "\u9E3F\uA60F\uAE07\uB61B\uBE1F\uC62B\uCE2D\uD611" +
                "\uDE03\uE5FF\uEEED\uF699\uFE3D\u071C\u0ED2\u1692" +
                "\u1E06\u26AE\u2EDC";

for (int iatwS = 0, qUJQG = 0; iatwS < 19; iatwS++)
{
        qUJQG = szTest.charAt(iatwS);
        qUJQG ++;
        qUJQG = ((qUJQG << 5) | ( (qUJQG & 0xFFFF) >> 11)) & 0xFFFF;
        qUJQG -= iatwS;
        qUJQG = (((qUJQG & 0xFFFF) >> 6) | (qUJQG << 10)) & 0xFFFF;
        qUJQG ^= iatwS;
        qUJQG -= iatwS;
        qUJQG = (((qUJQG & 0xFFFF) >> 3) | (qUJQG << 13)) & 0xFFFF;
        qUJQG ^= 0xFFFF;
        qUJQG ^= 0xB6EC;
        qUJQG = ((qUJQG << 8) | ( (qUJQG & 0xFFFF) >> 8)) & 0xFFFF;
        qUJQG --;
        qUJQG = (((qUJQG & 0xFFFF) >> 5) | (qUJQG << 11)) & 0xFFFF;
        qUJQG ++;
        qUJQG ^= 0xFFFF;
        qUJQG += iatwS;
        szTest = szTest.substring(0, iatwS) + (char)(qUJQG & 0xFFFF) + szTest.substring(iatwS + 1);
}

System.out.println(szTest);

우리 회사에서 항상 사용합니다.


이것은 모호함을 통한 보안 이며 실제로 안전하지는 않습니다.
Chloe

이 질문은 문자열을 정적으로 추출하기 어렵게 만드는 난독 화뿐만 아니라 AES와 같은 실제 최신 암호화 강도 암호화를 요구합니다. 이것은 문자 사이의 상태를 유지하지 않는 것처럼 보이므로 빈도 분석에 취약합니다. ( 라틴어 알파벳 대신 UTF-16 코드 포인트를 제외한 단일-알파벳 대체 암호 . 그러나 영어 ASCII 텍스트에서 사용하는 경우, 내가 잘못 읽지 않는 한 몇 가지 고유 한 16 비트 문자 값만 얻을 수 있습니다)
Peter Cordes

-4
String s1="arshad"; 
char[] s2=s1.toCharArray(); 
int s3= s2.length; 

  System.out.println(s3);
 int i=0; 

// for(int j=0;j<s3;j++) 
// System.out.println(s2[j]); 

for(i=0;i<((s3)/2);i++) { 

char z,f=10; 
z=(char) (s2[i] * f); 
s2[i]=s2[(s3-1)-i]; 
s2[(s3-1)-i]=z; 

String b=new String(s2);

 print(b);  }

공식적으로 데이터를 읽을 수없는 형식으로 암호화합니다. 암호를 해독하려면 동일한 코드를 사용하십시오. s [i] * f를 s [I] / f로 변경하십시오.
Arshad shaik

이것은 모호함을 통한 보안 이며 실제로 안전하지는 않습니다.
Chloe

-5
public static String encryptParams(String myTextInput) {

        String myKey = "40674244454045cb9a70040a30e1c007";
        String myVector = "@1B2c3D4e5F6g7H8";

        String encData = "";

        try{
            JavaEncryprtionUtil encUtil = new JavaEncryprtionUtil();
            encData = Base64.encodeToString(encUtil.encrypt(myTextInput.getBytes("UTF-8"), myKey.getBytes("UTF-8"), myVector.getBytes("UTF-8")),Base64.DEFAULT);
            System.out.println(encData);
        }catch(NoSuchAlgorithmException ex){
            ex.printStackTrace();
        }catch(NoSuchPaddingException ex){
            ex.printStackTrace();
        }catch(InvalidKeyException ex){
            ex.printStackTrace();
        }catch(InvalidAlgorithmParameterException ex){
            ex.printStackTrace();
        }catch(IllegalBlockSizeException ex){
            ex.printStackTrace();
        }catch(BadPaddingException ex){
            ex.printStackTrace();
        }catch(UnsupportedEncodingException ex){
            ex.printStackTrace();
        }

        return encData;
    }

1
JavaEncryprtionUtil은 JDK API의 일부입니까? 그렇지 않은 경우 라이브러리 이름을 입력해야합니다.
Fermat의 작은 학생

4
해당 수업을 찾을 수 없습니다. 답변이 완성 된 것 같습니다.
james.garriss
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.