DeflateStream and GZipStream directly inherit Stream class.
Since they work on streams, they read the data byte by byte. So they cannot determine the best compression based on the overall data. Since they are based on DEFLATE algorithm which works like a dictionary encoder, they are not good with already compressed data. Neither of them supports adding individual files to the compressed stream. Both of these types are supported in metro style (.net for Windows Store) apps and Portable Class Library projects.
DeflateStream and GZipStream were introduced in .net framework 2.0 to compress a stream. In .net framework 4.5, they are updated to use zlib library. Because of the changes in Stream, they also support asynchronous behavior.
Compression using DeflateStream
DeflateStream is based on DEFLATE algorithm. Compressing a stream is just about copying another stream to DeflateStream instance with the proper compression mode set. In the following code, the path of source and destination file is specified as parameters to a method. It just reads the source file using FileStream and compresses it using DeflateStream to destination path.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/// <summary> | |
/// Compress source file | |
/// </summary> | |
/// <param name="sourceFile">Source File Path</param> | |
/// <param name="destinationFile">Destination File Path</param> | |
private static void CompressFileUsingDeflateStream(string sourceFile, string destinationFile) | |
{ | |
using (var sourceFileStream = File.OpenRead(sourceFile)) | |
{ | |
using (var destinationFileStream = File.Create(destinationFile)) | |
{ | |
using (var deflateDestinationStream = new DeflateStream(destinationFileStream, CompressionMode.Compress)) | |
{ | |
sourceFileStream.CopyTo(deflateDestinationStream); | |
} | |
} | |
} | |
} |
Decompression is also as easier as compression. Here we are reading a source file using FileStream and passing it to DeflateStream with Decompression mode set. Then it is just a matter of copying it to a destination FileStream object. The term Inflate is used for decompression.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/// <summary> | |
/// Decompress file using Deflate Stream | |
/// </summary> | |
/// <param name="sourceFile">Source File</param> | |
/// <param name="destinationFile">Destination File</param> | |
private static void DecompressFileUsingDeflateStream(string sourceFile, string destinationFile) | |
{ | |
using (var sourceStream = File.OpenRead(sourceFile)) | |
{ | |
using (var deflateStream = new DeflateStream(sourceStream, CompressionMode.Decompress)) | |
{ | |
using (var desinationStream = File.Create(destinationFile)) | |
{ | |
deflateStream.CopyTo(desinationStream); | |
} | |
} | |
} | |
} |
GZipStream Compression
GZip uses an industry standards data format for compression. The compressed stream can be saved with a *.gz extension. This compressed file can be decompressed with commonly available decompression tools and utilities. Using GZipStream is similar to using DeflateStream. The following code reads a file using FileStream and compresses it using a GZipStream object.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
private static void CompressFileUsingGZipStream(string sourceFile, string destinationFile) | |
{ | |
using (var sourceFileStream = File.OpenRead(sourceFile)) | |
{ | |
using (var destinationFileStream = File.Create(destinationFile)) | |
{ | |
using (var gZipDestinationStream = new GZipStream(destinationFileStream, CompressionMode.Compress)) | |
{ | |
sourceFileStream.CopyTo(gZipDestinationStream); | |
} | |
} | |
} | |
} |
And the following is the code which can be used for decompressing a file using GZipStream. The file must have been compressed with GZip format.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
private static void DecompressFileUsingGZipStream(string sourceFile, string destinationFile) | |
{ | |
using (var sourceStream = File.OpenRead(sourceFile)) | |
{ | |
using (var deflateStream = new GZipStream(sourceStream, CompressionMode.Decompress)) | |
{ | |
using (var desinationStream = File.Create(destinationFile)) | |
{ | |
deflateStream.CopyTo(desinationStream); | |
} | |
} | |
} | |
} |
Async compression and decompression
GZipStream and DeflateStream supports two Microsoft Async patterns. They are as follows:
- Asynchronous Programming Model [APM] The compression streams support APM by providing BeginRead / EndRead and BeginWrite / EndWrite method pairs. This is a legacy asynchronous pattern by Microsoft and is no longer recommended for new development. The pattern is also sometimes known as ASynchResult pattern. The BeginX / EndX method pair starts and ends the asynchronous operations. The BeginX method returns an ASyncResult object which can be used with the corresponding EndX method to block until the operation started with BeginX is completed. The IASyncResult can also be used to wait on the asynchronous operation to complete. The exceptions are also rethrown in the calling thread when EndX operation is called. Although BeginRead / EndRead and BeginWrite / EndWrite method pairs are inherited from Stream. They do get overridden in GZipStream and DeflateStream but since APM is no more recommended after .net framework 4.5, these methods are not recommended to be used for new development.
- Task-based Asynchronous Pattern [TAP] .net framework 4.5 introduced new XAsync methods to support another Microsoft asynchronous pattern TAP [Task based Asynchronous Pattern]. There are following methods provided in .net framework 4.5:
- FlushAsync
- ReadAsync
- WriteAsync
- CopyToAsync
These methods, including their overrides, are inherited from Stream. There are overrides for these methods which support cancellation.
As we discussed, DeflateStream is based on LZ77 and Huffman coding. LZ77 work on dictionary based encoding. Based on the extra bits created for dictionary keys, they might result in bigger compressed data, even bigger than the source data. So if the data size is too small, then it is better not to use them at all.
Download Code