Разница между Convert.ToBase64String/Convert.FromBase64String и Encoding.UTF8.GetBytes/Encoding.UTF8.GetString

В настоящее время я изучаю Symmetric Cryptography в .NET. Я написал демо, как показано ниже:

    private byte[] key = Encoding.ASCII.GetBytes("abcdefgh");
    private byte[] IV = Encoding.ASCII.GetBytes("hgfedcba");
    private byte[] encrypted;

    public Form1()
    {
        InitializeComponent();

    }

    private void btnEncrypt_Click(object sender, EventArgs e)
    {
        this.textBox2.Text = this.Encrypt(this.textBox1.Text);
    }

    private void btnDecrypt_Click(object sender, EventArgs e)
    {
        this.textBox3.Text = this.Decrypt(this.textBox2.Text);
    }

    private string Encrypt(string plainText)
    {
        try
        {
            using (DESCryptoServiceProvider crypto = new DESCryptoServiceProvider())
            {
                crypto.Key = this.key;
                crypto.IV = this.IV;

                ICryptoTransform transform = crypto.CreateEncryptor(crypto.Key, crypto.IV);

                using (MemoryStream stream = new MemoryStream())
                {
                    using (CryptoStream cryptoStream = new CryptoStream(stream, transform, CryptoStreamMode.Write))
                    {
                        using (StreamWriter writer = new StreamWriter(cryptoStream))
                        {
                            writer.Write(plainText);
                        }

                        encrypted = stream.ToArray();
                    }
                }
            }

            return Convert.ToBase64String(encrypted);
        }
        catch (Exception)
        {

            throw;
        }
    }

    private string Decrypt(string cipherText)
    {
        try
        {
            string plainText = string.Empty;

            using (DESCryptoServiceProvider crypto = new DESCryptoServiceProvider())
            {
                crypto.Key = this.key;
                crypto.IV = this.IV;

                ICryptoTransform transform = crypto.CreateDecryptor(crypto.Key, crypto.IV);

                using (MemoryStream stream = new MemoryStream(Convert.FromBase64String(cipherText)))
                {
                    using (CryptoStream cryptoStream = new CryptoStream(stream, transform, CryptoStreamMode.Read))
                    {
                        using (StreamReader reader = new StreamReader(cryptoStream))
                        {
                            plainText = reader.ReadToEnd();
                        }
                    }
                }
            }

            return plainText;
        }
        catch (Exception)
        {

            throw;
        }
    }

Все работает так, как ожидалось. Но если я заменил

return Convert.ToBase64String(encrypted);

И

using (MemoryStream stream = new MemoryStream(Convert.FromBase64String(cipherText)))

Для

return Encoding.UTF8.GetString(encrypted);

И

using (MemoryStream stream = new MemoryStream(Encoding.UTF8.GetBytes(cipherText)))

В CryptoStream System.NotSupportedException появилась ошибка. После диагностики кода я обнаружил, что Encoding.UTF8.GetBytes(cipherText) имеет больше байтов, чем encrypted

В чем разница между использованием Convert.From/ToBase64String и Encoding.UTF8.GetBytes/GetString)?

Ответ 1

UTF-8 - это кодировка символов. Он кодирует коды Unicode (символы) в байты.

Base64 - двоично-текстовое кодирование. Он кодирует байты в текст.

В этом случае вам понадобится последний.

Encoding.UTF8.GetString декодирует байтовый массив с кодировкой UTF-8, который является потерянным, если имеются недопустимые последовательности байтов (что весьма вероятно, если вы кормите его случайными байтами, как зашифрованный текст).