1

有一个方法 ( Version1 ) 可以对输入流进行编码,并且有一个函数 Decrypt() 可以成功解码编码数据。但是当输入数据很大时,可能会出现错误OutOfMemory(在“ string textEncrypted = Convert.ToBase64String (ms.ToArray()) ”这一行上)。

版本1

    private static Stream EncryptRijndael1(byte[] key, byte[] iv, Stream plainText)
    {
        if (plainText == null)
            return null;

        byte[] bytesEncrypted;

        RijndaelManaged rjndl = RijndaelManagedWithConfig(key, iv);

        using (MemoryStream ms = new MemoryStream())
        {
            using (CryptoStream cs = new CryptoStream(ms, rjndl.CreateEncryptor(key, iv), CryptoStreamMode.Write))
            {
                byte[] buffer = new byte[16 * 1024];
                int readed;
                while ((readed = (plainText.Read(buffer, 0, buffer.Length))) > 0)
                {
                    cs.Write(buffer, 0, readed);
                }
            }
            string textEncrypted = Convert.ToBase64String(ms.ToArray());
            bytesEncrypted = Encoding.ASCII.GetBytes(textEncrypted);
        }
        return new MemoryStream(bytesEncrypted);
    }

所以我修改了方法来逐部分(块)处理数组。这是版本2它会在Convert.ToBase64String (ms.ToArray(), offset, read)行中导致错误“偏移量和长度必须引用字符串中的位置”

版本 2

    private static Stream EncryptRijndael2(byte[] key, byte[] iv, Stream plainText)
    {
        if (plainText == null)
            return null;

        byte[] bytesEncrypted;

        RijndaelManaged rjndl = RijndaelManagedWithConfig(key, iv);

        string textEncrypted = String.Empty;

        using (MemoryStream ms = new MemoryStream())
        {
            using (CryptoStream cs = new CryptoStream(ms, rjndl.CreateEncryptor(key, iv), CryptoStreamMode.Write))
            {
                byte[] buffer = new byte[16 * 1024];
                int readed;
                int offset = 0;
                while ((readed = (plainText.Read(buffer, 0, buffer.Length))) > 0)
                {
                    cs.Write(buffer, 0, readed);
                    textEncrypted += Convert.ToBase64String(ms.ToArray(), offset, readed);
                    offset += readed;
                }
            }

            bytesEncrypted = Encoding.ASCII.GetBytes(textEncrypted);

        }

        return new MemoryStream(bytesEncrypted);
    }

然后我做了Version3。它可以正常工作,但现在输出数据长度大于具有相同输入数据的版本 1。Decryt() 函数抛出错误“输入不是有效的 Base-64 字符串,因为它包含非 base 64 字符、两个以上的填充字符或填充字符中的非法字符。”

版本 3

    private static Stream EncryptRijndael3(byte[] key, byte[] iv, Stream plainText)
    {
        if (plainText == null)
            return null;

        byte[] bytesEncrypted;

        RijndaelManaged rjndl = RijndaelManagedWithConfig(key, iv);

        using (MemoryStream ms = new MemoryStream())
        {
            string textEncrypted = String.Empty;
            using (CryptoStream cs = new CryptoStream(ms, rjndl.CreateEncryptor(key, iv), CryptoStreamMode.Write))
            {
                byte[] buffer = new byte[16*1024];
                int readed;
                while ((readed = (plainText.Read(buffer, 0, buffer.Length))) > 0)
                {
                    cs.Write(buffer, 0, readed);
                }
             }


                byte[] buffer1 = new byte[16*1024];
                int readed1;
                using (MemoryStream ms1 = new MemoryStream(ms.ToArray()))
                {
                    while ((readed1 = (ms1.Read(buffer1, 0, buffer1.Length))) > 0)
                    {
                        if (readed1 < buffer1.Length)
                        {
                            var lastBuf = new List<Byte>();
                            for (int i = 0; i < readed1; i++)
                            {
                                lastBuf.Add(buffer1[i]);
                            }
                            textEncrypted += Convert.ToBase64String(lastBuf.ToArray());
                            continue;
                        }
                        textEncrypted += Convert.ToBase64String(buffer1);
                    }
                 }

            bytesEncrypted = Encoding.ASCII.GetBytes(textEncrypted);
        }

        return new MemoryStream(bytesEncrypted);
    }

我的 RijndaelManaged

 private static RijndaelManaged RijndaelManagedWithConfig(byte[] key, byte[] iv)
    {
        RijndaelManaged rjndl = new RijndaelManaged();
        rjndl.KeySize = 256;
        rjndl.BlockSize = 128;

        rjndl.Key = key;
        rjndl.IV = iv;

        rjndl.Mode = CipherMode.CBC;
        rjndl.Padding = PaddingMode.PKCS7;

        return rjndl;
    }

请帮助我摆脱错误或告诉我如何使 Version1 进程Convert.ToBase64String 部分数据

4

1 回答 1

0

我已经实现了决定

         private static Stream EncryptRijndael(byte[] key, byte[] iv, Stream plainText)
    {
        if (plainText == null)
            return null;

        byte[] buffer = new byte[5120 * 1024];

        RijndaelManaged rjndl = RijndaelManagedWithConfig(key, iv);

        using (var memoryStream = new MemoryStream())
        {
            int readedBytes;
            using (var cs = new CryptoStream(memoryStream, rjndl.CreateEncryptor(key, iv), CryptoStreamMode.Write))
            {
                while ((readedBytes = (plainText.Read(buffer, 0, buffer.Length))) > 0)
                {
                    cs.Write(buffer, 0, readedBytes);
                }
            }


            using (var cryptoMemoryStream = new MemoryStream(memoryStream.ToArray()))
            {
                using (var base64MemoryStream = new MemoryStream())
                {
                    using (ICryptoTransform transform = new ToBase64Transform())
                    {
                        using (var cryptStream = new CryptoStream(base64MemoryStream, transform, CryptoStreamMode.Write))
                        {
                            while ((readedBytes = cryptoMemoryStream.Read(buffer, 0, buffer.Length)) > 0)
                            {
                                cryptStream.Write(buffer, 0, readedBytes);
                            }
                            cryptStream.FlushFinalBlock();
                        }

                        return new MemoryStream(base64MemoryStream.ToArray());
                    }
                }
            }
        }
    }
于 2016-12-05T20:21:29.080 回答