Delphi 中压缩流和解压流的应用

本文介绍Delphi中TCompressionStream和TDecompressionStream类的使用方法,包括压缩和解压缩流程,以及如何在实际项目中应用这些技术。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 Delphi 中压缩流和解压流的应用

软件开发者不免都要遇到压缩数据的问题!经常使用Delphi的朋友都知道,它为我们提供了两个流类(TCompressionStream和TDecompressionStream)来完成数据的压缩和解压缩,但美中不足的是,该流在Delphi 的帮助中没有详细的说明,使得它们在使用起来有一定得困难。其实在Delphi系统中提供了这两个类的源代码和库。保存在Delphi 光盘的/Info/Extras/Zlib/ Src和/Info/Extras/Zlib/Obj目录中(其中OBJ录中保存的是库,Src目录中保存的是源代码,感兴趣的朋友可以看看)。本人在使用的过程中,对它们有了一定的了解。
    一、 类的说明
    1、 基类 TCustomZlibStream:类TCustomZlibStream 是类TCompressionStream和TDecompressionStream 类的基类,它主要有一个属性: OnProgress,在类进行压缩或解压缩的过程中会发生这个的事件 ,它的定义如下:
    Procedure OnProgress (Sender: TObject); dynamic;

    2、 压缩类TCompressionStream:类TCompressionStream除了继承了基类的 OnProgress 属性外,又增加了一个属性:CompressionRate,它的定义如下:
    Property CompressionRate: Single read GetCompressionRate;通过这个属性,可以得到压缩比。

    它的几个重要的方法定义如下:
    Constructor TCompressionStream.Create (CompressionLevel: TCompressionLevel; Dest: TStream);

    其中:TcompressionLevel(压缩类型),它由如下几个定义:
    ①、 clNone :不进行数据压缩;
    ②、 clFastest:进行快速压缩,牺牲压缩效率;
    ③、 clDefault:进行正常压缩;
    ④、 clMax: 进行最大化压缩,牺牲速度;
    Dest:目的流,用于存放压缩过的数据。

    Function TCompressionStream.Write (const Buffer; Count: Longint): Longint;
    其中:Buffer:需要压缩的数据;
    Count: 需要压缩的数据的字节数;
    函数返回写入流的字节数。
    压缩类TCompressionStream的数据只能是写入的,如果试图从其内部读取数据,将发生一个"Error "异常。需要压缩的数据通过方法 Write写入流中,在写入的过程中就被压缩,并保存在由构造函数提供的内存流(TmemoryStream)中,同时触发 OnProcess 事件。

    3、 解压缩类 TDecompressionStream :和压缩类TcompressionStream 相反,它的数据是只能读出的,如果试图往其内部写数据,将发生一个"Error "异常。它的几个重要方法定义如下:
    构造函数:Constructor Create(Source: TStream);
    其中:Source 是保存着压缩数据的流;
    Function Read(var Buffer; Count: Longint): Longint;
    数据读出函数,Buffer: 存数据缓冲区;
    Count: 缓冲区的大小;
    函数返回读出的字节数。
    数据在读出的过程中,数据被解压缩,并触发 OnProcess 事件。

    二、 类的使用
    通过类TCompressionStream和TdecompressionStream的配合使用,我们可以非常方便地完成数据的压缩和解压,下面就是本人在编写屏幕拷贝程序中的使用例子:

 

  1.     Procedure TClientForm.GetScreen;
  2.     Var
  3.     SourceDC,DestDC:HDC;
  4.     Bhandle:HBITMAP;
  5.     BitMap:TBitMap;
  6.     BmpStream,Deststream:TMemoryStream;
  7.     SourceStream:TCompressionStream;
  8.     Count:Integer;
  9.     Begin
  10.     SourceDC:=CreateDC('display','','',nil);
  11.     {得到屏幕的 DC}
  12.     DestDC:=CreateCompatibleDC(SourceDC);
  13.     {建立临时 DC}
  14.     Bhandle:=CreateCompatibleBitmap(SourceDC,Screen.Width, Screen.Height);
  15.     {建立位图}
  16.     SelectObject(DestDC,Bhandle);
  17.     {选择位图DC}
  18.     BitBlt(DestDC,0,0,Screen.Width, Screen.Height,SourceDC,0,0,SRCCOPY);
  19.     {拷贝整个屏幕}
  20.     BitMap:=TBitMap.Create;
  21.     BitMap.Handle := Bhandle;
  22.     {保存屏幕位图到 BitMap中}
  23.     BmpStream:=TMemoryStream.Create;
  24.     BitMap.SaveToStream(BmpStream);
  25.     {建立位图数据的内存流}
  26.     count:=BmpStream.Size;
  27.     {保存位图的大小}
  28.     DestStream:=TMemoryStream.Create;
  29.     {目标流,保存压缩数据}
  30.     SourceStream:=TCompressionStream.Create(clMax, DestStream);
  31.     {构建压缩流,采用最大化压缩,并保存到目标流中}
  32.     try
  33.     BmpStream.SaveToStream(SourceStream);
  34.     {压缩位图流}
  35.     SourceStream.Free;
  36.     {完成压缩,释放压缩流}
  37.     BmpStream.Clear;
  38.     {清空原来位图流}
  39.     BmpStream.WriteBuffer(Count, Sizeof(Count));
  40.     {将原来位图的大小保存到新的位图流中,以便使用}
  41.     BmpStream.CopyFrom(DestStream, 0);
  42.     {将压缩数据附加到新的位图流后面}
  43.     BmpStream.Position := 0;
  44.     NMStrm.PostIt(BmpStream);
  45.     {发送位图流}
  46.     finally
  47.     DestStream.Free;
  48.     BmpStream.Destroy ;
  49.     BitMap.Destroy;
  50.     DeleteDC(SourceDC);
  51.     ReleaseDC(Bhandle,SourceDC);
  52.     end; 
  53.     {释放有关资源}
  54.     End; 
  55. //    该过程得到整个屏幕的图象拷贝,并利用压缩流SourceStream和内存流 Deststream将位图压缩,并重新把位图大小和压缩数据流保存到位图流中,发送出去,发送位图大小的目的是在解压前来确定需要的内存空间。
  56.     procedure TServerForm.NMStrmServMSG(Sender: TComponent;
  57.     const sFrom: String; strm: TStream);
  58.     Var
  59.     StreamStr,DestStream:TMemoryStream;
  60.     SourceStream:TDecompressionStream;
  61.     count:Integer;
  62.     buffer:pointer;
  63.     begin
  64.     ScreenImage.Picture.Bitmap:=nil;
  65.     If Strm Is TMemoryStream Then
  66.     StreamStr := Strm AS TMemoryStream
  67.     Else
  68.     Exit;
  69.     StreamStr.Position := 0;
  70.     StreamStr.ReadBuffer(Count, Sizeof(Count));
  71.     {得到位图的大小}
  72.     GetMem(Buffer,Count);
  73.     {申请数据空间}
  74.     DestStream := TMemoryStream.Create;
  75.     SourceStream := TDecompressionStream.Create(StreamStr);
  76.     {构建解压流,压缩数据由StreamStr 流得到}
  77.     StatusBar.SimpleText := '正在处理图象';
  78.     Try
  79.     SourceStream.ReadBuffer(Buffer^,Count);
  80.     {读出解压数据}
  81.     DestStream.WriteBuffer(Buffer^,Count);
  82.     {保存到位图流中}
  83.     DestStream.Position := 0;
  84.     ScreenImage.Picture.Bitmap.LoadFromStream(DestStream);
  85.     {显示到屏幕上}
  86.     Finally
  87.     FreeMem(Buffer);
  88.     DestStream.Destroy;
  89.     SourceStream.Destroy;
  90.     End;
  91.     end;

    该过程首先从得到的数据流中取得位图大小,并申请内存空间,然后建立解压流,并将解压数据保存到位图流中,然后显示到屏幕上。
http://develop.csai.cn/delphi/200611151359101254.htm

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值