在.NET之前,使用非托管的Win32APIs加密解密数据是一件非常痛苦的事情。为了这个加密解密的目的,.NET配置了一组类(和命名空间)。现在你有很多类可以使用每种不同的算法保护你的数据。在.NET里面Crypttography命名空间下又定义了3种类型的加密方法。他们是AsymmetricAlgorithm,SymmetricAlgorithm和HashAlgorithm。所有的这些类(和.NET密码学类型)都是抽象类。我们今天将要描述SymmetricAlgorithm。剩下的将在以后的文章中陆续讲解。<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />
注意:虽然大多数托管代码里的加密类实现使用了很多CryptoAPI库。
对称算法使用用户的密钥(密码)工作。它的意思是,任何时候你都可以实现它而且可以使用对称算法加密或者解密你的数据,为了加密或者解密你的数据你必须定义一个密码或者一个密钥。下面描述了对称加密的特性。
另外在使用密钥或者密码对称加密过程中有一件非常重要的事情。就是初始化向量(IV)。IV被使用在最初的编码中(加密或者解密)。在所有的对称算法类中我们有一个名叫Mode的属性。这是被IV使用的。如果我们设置Mode属性为CipherMode.CBC(Cipher Block Chaining),则使用这个模式,每个数据块使用来自前一个块的值来处理。意思是如果系统在处理第三块数据,则它会从第二块中取一些信息(处理第三块数据)。接着它会取第一块数据中的信息用来处理第二块数据。但是在第一块数据之前没有可以用的块,因此它将使用IV来处理第一块。这个技术确保没有两个相同的块产生相同的输出并且因此使得数据更安全。然而如果你使Mode=CipherModer.ECB(Electronic codebook mode),则他不会使用上面的方法(使用前面的处理的块信息处理后面的块)。如果你想用很少的资源和时间处理大量的消息那么这个方法对于你来说就很有用。他也可以让你从数据的中间开始处理。
应此,上面我们包含了在对称加密中的两间非常重要的事情。他们是密钥和初始化向量。现在让我们看看对称加密支持哪些算法。
下面是对称算法和他们的类的关键信息。
算法名称 | 算法类(抽象) | 有效密钥大小(Bit) | 默认密钥大小(Bit) | 默认实现类 |
---|---|---|---|---|
DES | DES | 64 | 64 | DESCryptoServiceProvider |
TripleDES | TripleDES | 128, 192 | 192 | TripleDESCryptoServiceProvider |
RC2 | RC2 | 40-128 | 128 | RC2CryptoServiceProvider |
RijnDael | RijnDael | 128, 192, 256 | 256 | RijnDaelManaged |
这里需要注意的是所有的算法类都是继承于抽象类SymmetricAlgorithm。并且你可以看到每个类都支持不同的密钥大小。相同的情况下,他们也支持不同的初始化向量的大小。正如我刚才所说的他们所有的类都是抽象类,因此我们不能直接创建这些抽象类的任何实例。但是SymmetricAlgorithm类(也是抽象类)有一个共享的方法叫Create可以不用操心它是如何实现来创建一个类的具体实例。意思是,你可以通过下面的方式使用它。
RC2 mRC2 = RC2.Create();
它将为您返回一个RC2默认实现的一个实例,而不用去关心具体如何实现RC2这个类。如果你想在微软以后更新RC2类的实现还能共享代码(成为可能),这个技术是非常有用的。在那种情况下,你的代码将自动适应它们的改变并且正确的工作。或者可能在将来RC2类用托管代码写,你的代码依然可以接受它。在相同的情况下,你同样可以使用下面的语句。
RC2 mCrypto = SymmetricAlgorithm.Create(“RC2”);
这也可以给你返回一个RC2的对象(默认实现)。在这种情况下你要使用重裁Create方法用算法的名字设置参数来返回算法的对象。 这个Create方法来自SymmetricAlgorithm类,并且向我前面说的所有的使用对称算法的其它类都继承于SymmetricAlgorithm,因此你可以在上面所有的类里面找到Create这个方法。意思是如果你使用RC2.Create(“DES”)则它也能工作并且将返回一个DES的对象。但是不能使用RC2类得到DES对象。
上面的机制看起来很有用。我们可以用同样的方法使用我们自己的算法定义自己的类。但是要想这样,我们必须要对machine.config文件作一些小的改动。我在这里不详细描述。你可以参考Wrox关于密码学的书得到更多的信息。
现在让我们看看SymmetricAlgorithm类里面的一些方法和属性。
在写代码之前,让我们说几件对我们理解代码非常用帮助的事情。
SymmetricAlgorithm类的CreateEncryptor和CreateDecryptor方法返回ICryptoTransform对象。IcryptoTransform是一个想要处理数据块的类来实现的接口。这个过程可以是加密,解密,散列,基于64的编码和解码等等。这个接口的基本目的是完成数据处理分块(The basic purpose of this Interface is to perform Blockwize processing of data.)。你可以直接使用它的实例,但是在大多数情况下,为了方便,我们通过其他的名叫CryptoStream来完成。让我们看一个例子是如何使用它的。
DES mCrypt = new SymmetricAlgorithm.Create(“DES”);
ICryptoTransform mTransform = mCrypt.CreateEncryptot();
CreateEncryptor或者CreateDecryptor是两个重裁的方法。如果你没有任何参数传入其中,那么将使用默认的密钥和IV(使用SymmetricAlgoruthm类里面的GenerateKey和GenerateIV方法)。另一方面,你可以通过传入一个IV和密钥到CreateEncryptor和CreateDecryptor的对象中。以致加密和解密将使用我们自己定义的IV和密钥。
CryptoStream类通常被使用来读写数据同时也在读或者写的时候加密或者解密数据。它是简单的包装了一下原始流类Stream。It uses the buffered access taking all worries from you to manage buffer, block sizes, padding etc.你可以使用下面的代码得到它的实例。
DES mCrypt = SymmetricAlgorithm.Create(“DES”);
IcryptoTransform mTransform = mCrypt.CreateEncryptor();
CryptoStream mStream = new CryptoStream(fileStream,mTransform,CryptoStramMode.Read)
fileStream是请求从硬盘或者内存中读取数据的原始文件的流(或者是MemoryStream)。现在通过使用mStream对象和StreamReader/StreamWriter对象读写数据。当你将要读写时,你的加密解密信息将依赖IcryptoTransform对象。
现在我们有足够关于SymmetricAlgorithm的信息。最后,让我们看看将要编码和解码的代码片断。我假设你拥有一个包含txtData和命令按钮控件的窗体。在命令按钮的代码事件里写如下代码。这个代码将要加密TextBox里面的文本并用MessageBox显示,在将加密结果写回到TextBox中。
SymmetricAlgorithm mCryptProv;
MemoryStream mMemStr;
// 加密txtData中的数据,然后将加密结果用MessageBox显示并且回写到TextBox中
// 这里你可以配置任何.NET支持的类
DES mCryptProv = SymmetricAlgorithm.Create(“Rijndael”);
// 加密数据将要以流的形式存储在内存中因此我们需要内存Stream对象
mMemStr = new MemoryStream();
// 创建ICryptTransform对象.(在这里我们使用默认的密钥和初始向量).
ICryptTramsform mTransform = mCryptProv.CreateEncryptor();
CryptoStream mCSWriter = new CryptoStream(mMemStr,mTransform,CryptoStreamMode.Write);
StreamWriter mSWriter = StreamWriter(mCSWriter);
mSwriter.Writer(this.txtData.Text);
mSWriter.Flush();
mCSWriter.FlushFinalBlock();
有一件事情在这里需要注意我们没有在代码的任何地方使用IV和密钥。事实上,在我们在代码里面没有指定他们的时候.NET Framework将为我们自动产生。但是本文包含的例子代码使用的是用户指定的密钥和IV。我们将加密以后的数据使用MemoryStream写到内存中。现在让我们从内存中得到数据的代码。
// 数据已经写入内存但是我们需要回显它到TextBox和MessageBox中,因此要做下面的工作。
// 为接受数据创建字节数组。
byte[] mBytes = new byte[mMemStr.Length - 1];
mMemStr.Position = 0;
mMemStr.Read(mBytes,0,mMemStr.Length);
Text.UTF8Encoding mEnc = new Text.UTF8Encoding();
String mEncData = mEnc.GetString(mBytes);
MessageBox.Show(“加密数据为:/n”+mEncData);
This.txtData.Text = mEncData;
从字节转换为字符串必须要编码。在这里我使用了UTF8Encoding。最后,让我们将解密后的数据再次显示在MessageBox和TextBox中。
// 现在让我们从内存中得到解密后的数据
// 因为我们的数据在内存中,所以我们需要重新使用MemoryStream对象。
// 将内存点置0
mMemStr.Position = 0;
mTransform = mCryptProv.CreateDecryptor();
CryptoStream mCSReader = new CryptoStream(mMemStr,mTransform,CryptoStreamMode.Read);
StreamReader mStrReader = new StreamReader(mCSReader);
String mDecData = mStrReader.ReadToEnd();
MessageBox(“解密数据为:/n”+mDecData);
This.txtData.Text = mDecData;
这是所有的工作。解密那些数据我们使用了相同的内存流。为了能从流的其实部分读数据我们首先将它置为开始。然后我们用SymmetricAlgorithm对象的CreateDecryptor方法创建IcryptoTransform对象。我们在上面的代码中为了解密重复使用了对象(mMemStr)。你可以创建新的对象(使用新的变量)。然后我们为了从内存中读取数据需要StreamReader对象。While reading that it will also decrypt that data since we passed CryptoStream object during the creation of StreamReader object.
.NET为我们提供了一个非常好的托管途径保护我们的数据。我们可以使用.NET内建的一组类来加密我们的数据。虽然很多的类后台依然使用Crypto APIs技术,我们使用老的Crypto APIs没有任何问题。但是我们可以不用担心那些类的具体实现来安全的使用这些类。在后面的文章我将描述非对称加密算法的神话和用途。
该文章的例子代码让你可以选择一个算法来加密或者解密数据。而且它还让你指定你自己的IV和密钥。代码以两种方式工作。一种是TextBox,意思是你在TextBox中写些东西然后加密或者解密这些内容。第二种,你可以选择要加密或者解密的文件。
本文由来源 21aspnet,由 javajgs_com 整理编辑,其版权均为 21aspnet 所有,文章内容系作者个人观点,不代表 Java架构师必看 对观点赞同或支持。如需转载,请注明文章来源。