Sample code for 30+ languages & platforms
Unicode C++

Streaming Compression

See more Compression Examples

Compress and decompress using a stream.

Chilkat Unicode C++ Downloads

Unicode C++
#include <CkFileAccessW.h>
#include <CkCompressionW.h>
#include <CkStreamW.h>
#include <CkTaskW.h>

void ChilkatSample(void)
    {
    bool success = false;

    // This example assumes the Chilkat API to have been previously unlocked.
    // See Global Unlock Sample for sample code.

    CkFileAccessW fac;
    CkCompressionW compress;
    compress.put_Algorithm(L"deflate");

    CkStreamW streamC;

    // This example sets the source and sink of the stream to files.
    // A stream can have use other streams as a source or sink,
    // or the application can itself be the source/sink by directly
    // reading or writing a stream.  (See below for an example of this..)
    streamC.put_SourceFile(L"qa_data/hamlet.xml");
    streamC.put_SinkFile(L"qa_output/hamlet_compressed.dat");

    // Compress from source to sink.
    success = compress.CompressStream(streamC);
    if (success != true) {
        wprintf(L"%s\n",compress.lastErrorText());
        return;
    }

    wprintf(L"File-to-file deflate compression successful.\n");
    // Note: The FileSize method returns a signed 32-bit integer.  If the file is potentially larger than 2GB, call FileSizeStr instead to return
    // the size of the file as a string, then convert to an integer value.
    wprintf(L"Original size = %d\n",fac.FileSize(streamC.sourceFile()));
    wprintf(L"Compressed size = %d\n",fac.FileSize(streamC.sinkFile()));

    // Now do file-to-file decompression
    CkStreamW streamD;
    streamD.put_SourceFile(L"qa_output/hamlet_compressed.dat");
    streamD.put_SinkFile(L"qa_output/hamlet_restored.xml");

    // Decompress from source to sink.
    success = compress.DecompressStream(streamD);
    if (success != true) {
        wprintf(L"%s\n",compress.lastErrorText());
        return;
    }

    wprintf(L"File-to-file deflate decompression successful.\n");

    // Let's double-check to see that the files are equal in size and content:

    bool bFilesEqual = fac.FileContentsEqual(streamC.sourceFile(),streamD.sinkFile());
    if (bFilesEqual != true) {
        wprintf(L"The output file is not equal to the input file!\n");
    }
    else {
        wprintf(L"The file was successfully compressed and decompressed.\n");
    }

    // ---------------------------------------------------------------------
    // Now let's decompress again, but this time w/ the application reading
    // the decompressed data directly from a stream.
    CkStreamW streamA;

    streamA.put_SourceFile(L"qa_output/hamlet_compressed.dat");

    // Start decompressing in a background thread.
    CkTaskW *task = compress.DecompressStreamAsync(streamA);
    success = task->Run();

    // Read decompressed data from streamA:
    const wchar_t *decompressedText = 0;
    while ((streamA.get_EndOfStream() != true)) {
        if (streamA.get_DataAvailable() == true) {
            decompressedText = streamA.readString();
            wprintf(L"%s\n",decompressedText);
        }

    }

    // Let's make sure the background task finished and that the decompress was successful.
    // It should already be the case that the task is finished.
    while ((task->get_Finished() != true)) {
        task->SleepMs(20);
    }

    // The decompressor may have finished, but it is possible that data
    // remains to be flushed.  
    if (streamA.get_DataAvailable() == true) {
        decompressedText = streamA.readString();
        wprintf(L"%s\n",decompressedText);
    }

    // Did streamA succeed in reading the entire file?
    if (task->get_TaskSuccess() != true) {
        wprintf(L"async decompress failed:\n");
        wprintf(L"%s\n",task->resultErrorText());
        success = false;
    }

    wprintf(L"The async decompress was successful.\n");
    }