Java 实现 AES256-ECB-PKCS7Padding(无需修改策略文件)

import javax.crypto.BadPaddingException;
import javax.crypto.IllegalBlockSizeException;
import javax.crypto.NoSuchPaddingException;
import java.security.InvalidKeyException;
import java.security.Key;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;

public class AES {
    public static class Cipher {
        public static final int ENCRYPT_MODE = 1;
        public static final int DECRYPT_MODE = 2;
        public static final String AES_ECB_PKCS7 = "AES/ECB/PKCS7Padding";
        private int opmode;
        private AESBlock block;

        public static Cipher getInstance(String var0) throws NoSuchAlgorithmException, NoSuchPaddingException {
            if (AES_ECB_PKCS7.equals(var0)) {
                Cipher cipher = new Cipher();
                cipher.block = new AESBlock();
                return cipher;
            }
            throw new NoSuchAlgorithmException(var0);
        }

        public void init(int var1, Key var2) throws InvalidKeyException {
            this.opmode = var1;
            this.block.init(var1 == DECRYPT_MODE, var2.getEncoded());
        }

        public byte[] doFinal(byte[] var1) throws IllegalBlockSizeException, BadPaddingException {
            int blockSize = this.block.getBlockSize();
            if (this.opmode == ENCRYPT_MODE) {
                byte[] pad = PKCS7Padding.pad(var1, blockSize);
                byte[] dst = new byte[pad.length];
                for (int i = 0; i < pad.length; i += blockSize) {
                    this.block.encryptBlock(pad, i, dst, i);
                }
                return dst;
            } else {
                byte[] dst = new byte[var1.length];
                for (int i = 0; i < var1.length; i += blockSize) {
                    this.block.decryptBlock(var1, i, dst, i);
                }
                return PKCS7Padding.unpad(dst);
            }
        }
    }

    public static class PKCS7Padding {

        public static byte[] pad(byte[] var1, int blockSize) {
            int tail = blockSize - var1.length % blockSize;
            byte[] dst = new byte[var1.length + tail];
            System.arraycopy(var1, 0, dst, 0, var1.length);
            Arrays.fill(dst, var1.length, dst.length, (byte) (tail & 255));
            return dst;
        }

        public static byte[] unpad(byte[] var1) {
            int tail = var1[var1.length - 1];
            byte[] dst = new byte[var1.length - tail];
            System.arraycopy(var1, 0, dst, 0, dst.length);
            return dst;
        }
    }

    /**
     * Base on openjdk commit 85648e14f25d4448d51db001a4629154ea82b452.
     */
    public static final class AESBlock {
        private static final int AES_BLOCK_SIZE = 16;
        private static final int[] AES_KEYSIZES = new int[]{16, 24, 32};
        private static final byte[] S = new byte[256], Si = new byte[256];
        private static final int[] T1 = new int[256], T2 = new int[256], T3 = new int[256], T4 = new int[256], T5 = new int[256], T6 = new int[256], T7 = new int[256], T8 = new int[256];
        private static final int[] U1 = new int[256], U2 = new int[256], U3 = new int[256], U4 = new int[256];
        private static final byte[] rcon = new byte[30];
        private static int[] alog = new int[256], log = new int[256];

        static {
            int ROOT = 0x11B;
            int i, j = 0;
            alog[0] = 1;
            for (i = 1; i < 256; i++) {
                j = (alog[i - 1] << 1) ^ alog[i - 1];
                if ((j & 0x100) != 0) {
                    j ^= ROOT;
                }
                alog[i] = j;
            }
            for (i = 1; i < 255; i++) {
                log[alog[i]] = i;
            }
            byte[][] A = new byte[][]
                    {
                            {1, 1, 1, 1, 1, 0, 0, 0},
                            {0, 1, 1, 1, 1, 1, 0, 0},
                            {0, 0, 1, 1, 1, 1, 1, 0},
                            {0, 0, 0, 1, 1, 1, 1, 1},
                            {1, 0, 0, 0, 1, 1, 1, 1},
                            {1, 1, 0, 0, 0, 1, 1, 1},
                            {1, 1, 1, 0, 0, 0, 1, 1},
                            {1, 1, 1, 1, 0, 0, 0, 1}
                    };
            byte[] B = new byte[]{0, 1, 1, 0, 0, 0, 1, 1};
            int t;
            byte[][] box = new byte[256][8];
            box[1][7] = 1;
            for (i = 2; i < 256; i++) {
                j = alog[255 - log[i]];
                for (t = 0; t < 8; t++) {
                    box[i][t] = (byte) ((j >>> (7 - t)) & 0x01);
                }
            }
            byte[][] cox = new byte[256][8];
            for (i = 0; i < 256; i++) {
                for (t = 0; t < 8; t++) {
                    cox[i][t] = B[t];
                    for (j = 0; j < 8; j++) {
                        cox[i][t] ^= A[t][j] * box[i][j];
                    }
                }
            }
            for (i = 0; i < 256; i++) {
                S[i] = (byte) (cox[i][0] << 7);
                for (t = 1; t < 8; t++) {
                    S[i] ^= cox[i][t] << (7 - t);
                }
                Si[S[i] & 0xFF] = (byte) i;
            }
            byte[][] G = new byte[][]{
                    {2, 1, 1, 3},
                    {3, 2, 1, 1},
                    {1, 3, 2, 1},
                    {1, 1, 3, 2}
            };
            byte[][] AA = new byte[4][8];
            for (i = 0; i < 4; i++) {
                for (j = 0; j < 4; j++) AA[i][j] = G[i][j];
                AA[i][i + 4] = 1;
            }
            byte pivot, tmp;
            byte[][] iG = new byte[4][4];
            for (i = 0; i < 4; i++) {
                pivot = AA[i][i];
                if (pivot == 0) {
                    t = i + 1;
                    while (AA[t][i] == 0) {
                        t++;
                    }
                    for (j = 0; j < 8; j++) {
                        tmp = AA[i][j];
                        AA[i][j] = AA[t][j];
                        AA[t][j] = tmp;
                    }
                    pivot = AA[i][i];
                }
                for (j = 0; j < 8; j++) {
                    if (AA[i][j] != 0) {
                        AA[i][j] = (byte)
                                alog[(255 + log[AA[i][j] & 0xFF] - log[pivot & 0xFF])
                                        % 255];
                    }
                }
                for (t = 0; t < 4; t++) {
                    if (i != t) {
                        for (j = i + 1; j < 8; j++) {
                            AA[t][j] ^= mul(AA[i][j], AA[t][i]);
                        }
                        AA[t][i] = 0;
                    }
                }
            }
            for (i = 0; i < 4; i++) {
                for (j = 0; j < 4; j++) {
                    iG[i][j] = AA[i][j + 4];
                }
            }
            int s;
            for (t = 0; t < 256; t++) {
                s = S[t];
                T1[t] = mul4(s, G[0]);
                T2[t] = mul4(s, G[1]);
                T3[t] = mul4(s, G[2]);
                T4[t] = mul4(s, G[3]);

                s = Si[t];
                T5[t] = mul4(s, iG[0]);
                T6[t] = mul4(s, iG[1]);
                T7[t] = mul4(s, iG[2]);
                T8[t] = mul4(s, iG[3]);

                U1[t] = mul4(t, iG[0]);
                U2[t] = mul4(t, iG[1]);
                U3[t] = mul4(t, iG[2]);
                U4[t] = mul4(t, iG[3]);
            }
            rcon[0] = 1;
            int r = 1;
            for (t = 1; t < 30; t++) {
                r = mul(2, r);
                rcon[t] = (byte) r;
            }
            log = null;
            alog = null;
        }

        private boolean ROUNDS_12 = false;
        private boolean ROUNDS_14 = false;
        private int[][] sessionK = null;
        private int[] K = null;
        private byte[] lastKey = null;
        private int limit = 0;

        AESBlock() {
        }

        private static int[] expandToSubKey(int[][] kr, boolean decrypting) {
            int total = kr.length;
            int[] expK = new int[total * 4];
            if (decrypting) {
                System.arraycopy(kr[total - 1], 0, expK, 0, 4);
                for (int i = 1; i < total; i++) {
                    System.arraycopy(kr[i - 1], 0, expK, i * 4, 4);
                }
            } else {
                for (int i = 0; i < total; i++) {
                    System.arraycopy(kr[i], 0, expK, i * 4, 4);
                }
            }
            return expK;
        }

        private static int mul(int a, int b) {
            return (a != 0 && b != 0) ?
                    alog[(log[a & 0xFF] + log[b & 0xFF]) % 255] :
                    0;
        }

        private static int mul4(int a, byte[] b) {
            if (a == 0) return 0;
            a = log[a & 0xFF];
            int a0 = (b[0] != 0) ? alog[(a + log[b[0] & 0xFF]) % 255] & 0xFF : 0;
            int a1 = (b[1] != 0) ? alog[(a + log[b[1] & 0xFF]) % 255] & 0xFF : 0;
            int a2 = (b[2] != 0) ? alog[(a + log[b[2] & 0xFF]) % 255] & 0xFF : 0;
            int a3 = (b[3] != 0) ? alog[(a + log[b[3] & 0xFF]) % 255] & 0xFF : 0;
            return a0 << 24 | a1 << 16 | a2 << 8 | a3;
        }

        private static boolean isKeySizeValid(int len) {
            for (int aesKeysize : AES_KEYSIZES) {
                if (len == aesKeysize) {
                    return false;
                }
            }
            return true;
        }

        private static int getRounds(int keySize) {
            return (keySize >> 2) + 6;
        }

        public int getBlockSize() {
            return AES_BLOCK_SIZE;
        }

        void init(boolean decrypting, byte[] key)
                throws InvalidKeyException {
            if (isKeySizeValid(key.length)) {
                throw new InvalidKeyException("Invalid AES key length: " + key.length + " bytes");
            }
            if (!MessageDigest.isEqual(key, lastKey)) {
                makeSessionKey(key);
                lastKey = key.clone();
            }
            this.K = sessionK[(decrypting ? 1 : 0)];
        }

        public void encryptBlock(byte[] in, int inOffset, byte[] out, int outOffset) {
            int keyOffset = 0;
            int t0 = ((in[inOffset++]) << 24 | (in[inOffset++] & 0xFF) << 16 | (in[inOffset++] & 0xFF) << 8 | (in[inOffset++] & 0xFF)) ^ K[keyOffset++];
            int t1 = ((in[inOffset++]) << 24 | (in[inOffset++] & 0xFF) << 16 | (in[inOffset++] & 0xFF) << 8 | (in[inOffset++] & 0xFF)) ^ K[keyOffset++];
            int t2 = ((in[inOffset++]) << 24 | (in[inOffset++] & 0xFF) << 16 | (in[inOffset++] & 0xFF) << 8 | (in[inOffset++] & 0xFF)) ^ K[keyOffset++];
            int t3 = ((in[inOffset++]) << 24 | (in[inOffset++] & 0xFF) << 16 | (in[inOffset++] & 0xFF) << 8 | (in[inOffset++] & 0xFF)) ^ K[keyOffset++];
            while (keyOffset < limit) {
                int a0, a1, a2;
                a0 = T1[(t0 >>> 24)] ^ T2[(t1 >>> 16) & 0xFF] ^ T3[(t2 >>> 8) & 0xFF] ^ T4[(t3) & 0xFF] ^ K[keyOffset++];
                a1 = T1[(t1 >>> 24)] ^ T2[(t2 >>> 16) & 0xFF] ^ T3[(t3 >>> 8) & 0xFF] ^ T4[(t0) & 0xFF] ^ K[keyOffset++];
                a2 = T1[(t2 >>> 24)] ^ T2[(t3 >>> 16) & 0xFF] ^ T3[(t0 >>> 8) & 0xFF] ^ T4[(t1) & 0xFF] ^ K[keyOffset++];
                t3 = T1[(t3 >>> 24)] ^ T2[(t0 >>> 16) & 0xFF] ^ T3[(t1 >>> 8) & 0xFF] ^ T4[(t2) & 0xFF] ^ K[keyOffset++];
                t0 = a0;
                t1 = a1;
                t2 = a2;
            }
            int tt = K[keyOffset++];
            out[outOffset++] = (byte) (S[(t0 >>> 24)] ^ (tt >>> 24));
            out[outOffset++] = (byte) (S[(t1 >>> 16) & 0xFF] ^ (tt >>> 16));
            out[outOffset++] = (byte) (S[(t2 >>> 8) & 0xFF] ^ (tt >>> 8));
            out[outOffset++] = (byte) (S[(t3) & 0xFF] ^ (tt));
            tt = K[keyOffset++];
            out[outOffset++] = (byte) (S[(t1 >>> 24)] ^ (tt >>> 24));
            out[outOffset++] = (byte) (S[(t2 >>> 16) & 0xFF] ^ (tt >>> 16));
            out[outOffset++] = (byte) (S[(t3 >>> 8) & 0xFF] ^ (tt >>> 8));
            out[outOffset++] = (byte) (S[(t0) & 0xFF] ^ (tt));
            tt = K[keyOffset++];
            out[outOffset++] = (byte) (S[(t2 >>> 24)] ^ (tt >>> 24));
            out[outOffset++] = (byte) (S[(t3 >>> 16) & 0xFF] ^ (tt >>> 16));
            out[outOffset++] = (byte) (S[(t0 >>> 8) & 0xFF] ^ (tt >>> 8));
            out[outOffset++] = (byte) (S[(t1) & 0xFF] ^ (tt));
            tt = K[keyOffset++];
            out[outOffset++] = (byte) (S[(t3 >>> 24)] ^ (tt >>> 24));
            out[outOffset++] = (byte) (S[(t0 >>> 16) & 0xFF] ^ (tt >>> 16));
            out[outOffset++] = (byte) (S[(t1 >>> 8) & 0xFF] ^ (tt >>> 8));
            out[outOffset] = (byte) (S[(t2) & 0xFF] ^ (tt));
        }

        public void decryptBlock(byte[] in, int inOffset, byte[] out, int outOffset) {
            int keyOffset = 4;
            int t0 = ((in[inOffset++]) << 24 | (in[inOffset++] & 0xFF) << 16 | (in[inOffset++] & 0xFF) << 8 | (in[inOffset++] & 0xFF)) ^ K[keyOffset++];
            int t1 = ((in[inOffset++]) << 24 | (in[inOffset++] & 0xFF) << 16 | (in[inOffset++] & 0xFF) << 8 | (in[inOffset++] & 0xFF)) ^ K[keyOffset++];
            int t2 = ((in[inOffset++]) << 24 | (in[inOffset++] & 0xFF) << 16 | (in[inOffset++] & 0xFF) << 8 | (in[inOffset++] & 0xFF)) ^ K[keyOffset++];
            int t3 = ((in[inOffset++]) << 24 | (in[inOffset++] & 0xFF) << 16 | (in[inOffset++] & 0xFF) << 8 | (in[inOffset] & 0xFF)) ^ K[keyOffset++];
            int a0, a1, a2;
            if (ROUNDS_12) {
                a0 = T5[(t0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(t2 >>> 8) & 0xFF] ^ T8[(t1) & 0xFF] ^ K[keyOffset++];
                a1 = T5[(t1 >>> 24)] ^ T6[(t0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(t2) & 0xFF] ^ K[keyOffset++];
                a2 = T5[(t2 >>> 24)] ^ T6[(t1 >>> 16) & 0xFF] ^ T7[(t0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
                t3 = T5[(t3 >>> 24)] ^ T6[(t2 >>> 16) & 0xFF] ^ T7[(t1 >>> 8) & 0xFF] ^ T8[(t0) & 0xFF] ^ K[keyOffset++];
                t0 = T5[(a0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(a2 >>> 8) & 0xFF] ^ T8[(a1) & 0xFF] ^ K[keyOffset++];
                t1 = T5[(a1 >>> 24)] ^ T6[(a0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(a2) & 0xFF] ^ K[keyOffset++];
                t2 = T5[(a2 >>> 24)] ^ T6[(a1 >>> 16) & 0xFF] ^ T7[(a0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
                t3 = T5[(t3 >>> 24)] ^ T6[(a2 >>> 16) & 0xFF] ^ T7[(a1 >>> 8) & 0xFF] ^ T8[(a0) & 0xFF] ^ K[keyOffset++];
                if (ROUNDS_14) {
                    a0 = T5[(t0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(t2 >>> 8) & 0xFF] ^ T8[(t1) & 0xFF] ^ K[keyOffset++];
                    a1 = T5[(t1 >>> 24)] ^ T6[(t0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(t2) & 0xFF] ^ K[keyOffset++];
                    a2 = T5[(t2 >>> 24)] ^ T6[(t1 >>> 16) & 0xFF] ^ T7[(t0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
                    t3 = T5[(t3 >>> 24)] ^ T6[(t2 >>> 16) & 0xFF] ^ T7[(t1 >>> 8) & 0xFF] ^ T8[(t0) & 0xFF] ^ K[keyOffset++];
                    t0 = T5[(a0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(a2 >>> 8) & 0xFF] ^ T8[(a1) & 0xFF] ^ K[keyOffset++];
                    t1 = T5[(a1 >>> 24)] ^ T6[(a0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(a2) & 0xFF] ^ K[keyOffset++];
                    t2 = T5[(a2 >>> 24)] ^ T6[(a1 >>> 16) & 0xFF] ^ T7[(a0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
                    t3 = T5[(t3 >>> 24)] ^ T6[(a2 >>> 16) & 0xFF] ^ T7[(a1 >>> 8) & 0xFF] ^ T8[(a0) & 0xFF] ^ K[keyOffset++];
                }
            }
            a0 = T5[(t0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(t2 >>> 8) & 0xFF] ^ T8[(t1) & 0xFF] ^ K[keyOffset++];
            a1 = T5[(t1 >>> 24)] ^ T6[(t0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(t2) & 0xFF] ^ K[keyOffset++];
            a2 = T5[(t2 >>> 24)] ^ T6[(t1 >>> 16) & 0xFF] ^ T7[(t0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
            t3 = T5[(t3 >>> 24)] ^ T6[(t2 >>> 16) & 0xFF] ^ T7[(t1 >>> 8) & 0xFF] ^ T8[(t0) & 0xFF] ^ K[keyOffset++];
            t0 = T5[(a0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(a2 >>> 8) & 0xFF] ^ T8[(a1) & 0xFF] ^ K[keyOffset++];
            t1 = T5[(a1 >>> 24)] ^ T6[(a0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(a2) & 0xFF] ^ K[keyOffset++];
            t2 = T5[(a2 >>> 24)] ^ T6[(a1 >>> 16) & 0xFF] ^ T7[(a0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
            t3 = T5[(t3 >>> 24)] ^ T6[(a2 >>> 16) & 0xFF] ^ T7[(a1 >>> 8) & 0xFF] ^ T8[(a0) & 0xFF] ^ K[keyOffset++];
            a0 = T5[(t0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(t2 >>> 8) & 0xFF] ^ T8[(t1) & 0xFF] ^ K[keyOffset++];
            a1 = T5[(t1 >>> 24)] ^ T6[(t0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(t2) & 0xFF] ^ K[keyOffset++];
            a2 = T5[(t2 >>> 24)] ^ T6[(t1 >>> 16) & 0xFF] ^ T7[(t0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
            t3 = T5[(t3 >>> 24)] ^ T6[(t2 >>> 16) & 0xFF] ^ T7[(t1 >>> 8) & 0xFF] ^ T8[(t0) & 0xFF] ^ K[keyOffset++];
            t0 = T5[(a0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(a2 >>> 8) & 0xFF] ^ T8[(a1) & 0xFF] ^ K[keyOffset++];
            t1 = T5[(a1 >>> 24)] ^ T6[(a0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(a2) & 0xFF] ^ K[keyOffset++];
            t2 = T5[(a2 >>> 24)] ^ T6[(a1 >>> 16) & 0xFF] ^ T7[(a0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
            t3 = T5[(t3 >>> 24)] ^ T6[(a2 >>> 16) & 0xFF] ^ T7[(a1 >>> 8) & 0xFF] ^ T8[(a0) & 0xFF] ^ K[keyOffset++];
            a0 = T5[(t0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(t2 >>> 8) & 0xFF] ^ T8[(t1) & 0xFF] ^ K[keyOffset++];
            a1 = T5[(t1 >>> 24)] ^ T6[(t0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(t2) & 0xFF] ^ K[keyOffset++];
            a2 = T5[(t2 >>> 24)] ^ T6[(t1 >>> 16) & 0xFF] ^ T7[(t0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
            t3 = T5[(t3 >>> 24)] ^ T6[(t2 >>> 16) & 0xFF] ^ T7[(t1 >>> 8) & 0xFF] ^ T8[(t0) & 0xFF] ^ K[keyOffset++];
            t0 = T5[(a0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(a2 >>> 8) & 0xFF] ^ T8[(a1) & 0xFF] ^ K[keyOffset++];
            t1 = T5[(a1 >>> 24)] ^ T6[(a0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(a2) & 0xFF] ^ K[keyOffset++];
            t2 = T5[(a2 >>> 24)] ^ T6[(a1 >>> 16) & 0xFF] ^ T7[(a0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
            t3 = T5[(t3 >>> 24)] ^ T6[(a2 >>> 16) & 0xFF] ^ T7[(a1 >>> 8) & 0xFF] ^ T8[(a0) & 0xFF] ^ K[keyOffset++];
            a0 = T5[(t0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(t2 >>> 8) & 0xFF] ^ T8[(t1) & 0xFF] ^ K[keyOffset++];
            a1 = T5[(t1 >>> 24)] ^ T6[(t0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(t2) & 0xFF] ^ K[keyOffset++];
            a2 = T5[(t2 >>> 24)] ^ T6[(t1 >>> 16) & 0xFF] ^ T7[(t0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
            t3 = T5[(t3 >>> 24)] ^ T6[(t2 >>> 16) & 0xFF] ^ T7[(t1 >>> 8) & 0xFF] ^ T8[(t0) & 0xFF] ^ K[keyOffset++];
            t0 = T5[(a0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(a2 >>> 8) & 0xFF] ^ T8[(a1) & 0xFF] ^ K[keyOffset++];
            t1 = T5[(a1 >>> 24)] ^ T6[(a0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(a2) & 0xFF] ^ K[keyOffset++];
            t2 = T5[(a2 >>> 24)] ^ T6[(a1 >>> 16) & 0xFF] ^ T7[(a0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
            t3 = T5[(t3 >>> 24)] ^ T6[(a2 >>> 16) & 0xFF] ^ T7[(a1 >>> 8) & 0xFF] ^ T8[(a0) & 0xFF] ^ K[keyOffset++];
            a0 = T5[(t0 >>> 24)] ^ T6[(t3 >>> 16) & 0xFF] ^ T7[(t2 >>> 8) & 0xFF] ^ T8[(t1) & 0xFF] ^ K[keyOffset++];
            a1 = T5[(t1 >>> 24)] ^ T6[(t0 >>> 16) & 0xFF] ^ T7[(t3 >>> 8) & 0xFF] ^ T8[(t2) & 0xFF] ^ K[keyOffset++];
            a2 = T5[(t2 >>> 24)] ^ T6[(t1 >>> 16) & 0xFF] ^ T7[(t0 >>> 8) & 0xFF] ^ T8[(t3) & 0xFF] ^ K[keyOffset++];
            t3 = T5[(t3 >>> 24)] ^ T6[(t2 >>> 16) & 0xFF] ^ T7[(t1 >>> 8) & 0xFF] ^ T8[(t0) & 0xFF] ^ K[keyOffset++];
            t1 = K[0];
            out[outOffset++] = (byte) (Si[(a0 >>> 24)] ^ (t1 >>> 24));
            out[outOffset++] = (byte) (Si[(t3 >>> 16) & 0xFF] ^ (t1 >>> 16));
            out[outOffset++] = (byte) (Si[(a2 >>> 8) & 0xFF] ^ (t1 >>> 8));
            out[outOffset++] = (byte) (Si[(a1) & 0xFF] ^ (t1));
            t1 = K[1];
            out[outOffset++] = (byte) (Si[(a1 >>> 24)] ^ (t1 >>> 24));
            out[outOffset++] = (byte) (Si[(a0 >>> 16) & 0xFF] ^ (t1 >>> 16));
            out[outOffset++] = (byte) (Si[(t3 >>> 8) & 0xFF] ^ (t1 >>> 8));
            out[outOffset++] = (byte) (Si[(a2) & 0xFF] ^ (t1));
            t1 = K[2];
            out[outOffset++] = (byte) (Si[(a2 >>> 24)] ^ (t1 >>> 24));
            out[outOffset++] = (byte) (Si[(a1 >>> 16) & 0xFF] ^ (t1 >>> 16));
            out[outOffset++] = (byte) (Si[(a0 >>> 8) & 0xFF] ^ (t1 >>> 8));
            out[outOffset++] = (byte) (Si[(t3) & 0xFF] ^ (t1));
            t1 = K[3];
            out[outOffset++] = (byte) (Si[(t3 >>> 24)] ^ (t1 >>> 24));
            out[outOffset++] = (byte) (Si[(a2 >>> 16) & 0xFF] ^ (t1 >>> 16));
            out[outOffset++] = (byte) (Si[(a1 >>> 8) & 0xFF] ^ (t1 >>> 8));
            out[outOffset] = (byte) (Si[(a0) & 0xFF] ^ (t1));
        }

        private void makeSessionKey(byte[] k) throws InvalidKeyException {
            if (k == null) {
                throw new InvalidKeyException("Empty key");
            }
            if (isKeySizeValid(k.length)) {
                throw new InvalidKeyException("Invalid AES key length: " + k.length + " bytes");
            }
            int ROUNDS = getRounds(k.length);
            int ROUND_KEY_COUNT = (ROUNDS + 1) * 4;
            int BC = 4;
            int[][] Ke = new int[ROUNDS + 1][4];
            int[][] Kd = new int[ROUNDS + 1][4];
            int KC = k.length / 4;
            int[] tk = new int[KC];
            int i, j;
            for (i = 0, j = 0; i < KC; i++, j += 4) {
                tk[i] = (k[j]) << 24 | (k[j + 1] & 0xFF) << 16 | (k[j + 2] & 0xFF) << 8 | (k[j + 3] & 0xFF);
            }
            int t = 0;
            for (j = 0; (j < KC) && (t < ROUND_KEY_COUNT); j++, t++) {
                Ke[t / 4][t % 4] = tk[j];
                Kd[ROUNDS - (t / 4)][t % 4] = tk[j];
            }
            int tt, rconpointer = 0;
            while (t < ROUND_KEY_COUNT) {
                tt = tk[KC - 1];
                tk[0] ^= (S[(tt >>> 16) & 0xFF]) << 24 ^ (S[(tt >>> 8) & 0xFF] & 0xFF) << 16 ^ (S[(tt) & 0xFF] & 0xFF) << 8 ^ (S[(tt >>> 24)] & 0xFF) ^ (rcon[rconpointer++]) << 24;
                if (KC != 8)
                    for (i = 1, j = 0; i < KC; i++, j++) tk[i] ^= tk[j];
                else {
                    for (i = 1, j = 0; i < KC / 2; i++, j++) tk[i] ^= tk[j];
                    tt = tk[KC / 2 - 1];
                    tk[KC / 2] ^= (S[(tt) & 0xFF] & 0xFF) ^ (S[(tt >>> 8) & 0xFF] & 0xFF) << 8 ^ (S[(tt >>> 16) & 0xFF] & 0xFF) << 16 ^ (S[(tt >>> 24)]) << 24;
                    for (j = KC / 2, i = j + 1; i < KC; i++, j++) tk[i] ^= tk[j];
                }
                for (j = 0; (j < KC) && (t < ROUND_KEY_COUNT); j++, t++) {
                    Ke[t / 4][t % 4] = tk[j];
                    Kd[ROUNDS - (t / 4)][t % 4] = tk[j];
                }
            }
            for (int r = 1; r < ROUNDS; r++) {
                for (j = 0; j < BC; j++) {
                    tt = Kd[r][j];
                    Kd[r][j] = U1[(tt >>> 24) & 0xFF] ^ U2[(tt >>> 16) & 0xFF] ^ U3[(tt >>> 8) & 0xFF] ^ U4[tt & 0xFF];
                }
            }
            int[] expandedKe = expandToSubKey(Ke, false);
            int[] expandedKd = expandToSubKey(Kd, true);
            ROUNDS_12 = (ROUNDS >= 12);
            ROUNDS_14 = (ROUNDS == 14);
            limit = ROUNDS * 4;
            sessionK = new int[][]{expandedKe, expandedKd};
        }
    }
}

测试用例

import javax.crypto.BadPaddingException;
import javax.crypto.Cipher;
import javax.crypto.IllegalBlockSizeException;
import javax.crypto.spec.SecretKeySpec;
import java.security.Key;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.util.Arrays;
import java.util.concurrent.ThreadLocalRandom;

/**
 * Created by Demon on 2020/7/23
 */
public class Main {
    public static void main(String[] args) throws Exception {
        int[] keySize = {16, 24, 32};
        for (int size : keySize) {
            for (int i = 0; i < 1000; i++) {
                Key key = genKey(size);
                OriCipher oriCipher = new OriCipher(key);
                NewCipher newCipher = new NewCipher(key);
                for (int j = 0; j < 1000; j++) {
                    byte[] bytes = genData();
                    around(oriCipher, newCipher, bytes);
                }
            }
        }
    }

    private static Key genKey(int kLen) throws NoSuchAlgorithmException {
        SecureRandom secureRandom = SecureRandom.getInstance("SHA1PRNG");
        byte[] bytes = new byte[kLen];
        secureRandom.nextBytes(bytes);
        return new SecretKeySpec(bytes, "AES");
    }

    private static byte[] genData() throws NoSuchAlgorithmException {
        int length = ThreadLocalRandom.current().nextInt(2048);
        byte[] bytes = new byte[length];
        ThreadLocalRandom.current().nextBytes(bytes);
        return bytes;
    }

    private static void around(OriCipher ori, NewCipher newc, byte[] src) throws BadPaddingException, IllegalBlockSizeException {
        byte[] encrypt = ori.encrypt(src);
        byte[] decrypt = newc.decrypt(encrypt);
        byte[] encrypt1 = newc.encrypt(decrypt);
        byte[] decrypt1 = ori.decrypt(encrypt1);
        assertEquals(src, decrypt1);
    }

    private static void assertEquals(byte[] b1, byte[] b2) {
        if (!Arrays.equals(b1, b2)) {
            throw new RuntimeException("\n" +
                    Arrays.toString(b1) +
                    "\n" +
                    Arrays.toString(b2) +
                    "\n");

        }
    }
}

class OriCipher {
    private Cipher cipher1;
    private Cipher cipher2;

    public OriCipher(Key key) {
        try {
            this.cipher1 = Cipher.getInstance("AES/ECB/PKCS5Padding");
            this.cipher1.init(Cipher.ENCRYPT_MODE, key);
            this.cipher2 = Cipher.getInstance("AES/ECB/PKCS5Padding");
            this.cipher2.init(Cipher.DECRYPT_MODE, key);
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    public byte[] encrypt(byte[] src) throws BadPaddingException, IllegalBlockSizeException {
        return this.cipher1.doFinal(src);
    }

    public byte[] decrypt(byte[] enc) throws BadPaddingException, IllegalBlockSizeException {
        return this.cipher2.doFinal(enc);
    }
}

class NewCipher {
    private AES.Cipher cipher1;
    private AES.Cipher cipher2;

    public NewCipher(Key key) {
        try {
            this.cipher1 = AES.Cipher.getInstance("AES/ECB/PKCS7Padding");
            this.cipher1.init(AES.Cipher.ENCRYPT_MODE, key);
            this.cipher2 = AES.Cipher.getInstance("AES/ECB/PKCS7Padding");
            this.cipher2.init(AES.Cipher.DECRYPT_MODE, key);
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    public byte[] encrypt(byte[] src) throws BadPaddingException, IllegalBlockSizeException {
        return this.cipher1.doFinal(src);
    }

    public byte[] decrypt(byte[] enc) throws BadPaddingException, IllegalBlockSizeException {
        return this.cipher2.doFinal(enc);
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,776评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,527评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,361评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,430评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,511评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,544评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,561评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,315评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,763评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,070评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,235评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,911评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,554评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,173评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,424评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,106评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,103评论 2 352