压缩屏幕拷贝

Borland公司推出的 RAD开发工具 Delphi 5.0作为 Windows平台上的主流开发工具,其可视化的开发环境和面向对象编程的强大功能已经吸引了无数的开发人员。但是,一些程序员在实际的开发过程中却时常为对大量的数据进行压缩而伤透脑筋,不得不去查找一些高效的压缩算法或在网上查找第三方的控件来实现压缩。难道 Delphi本身没有提供这个功能吗?其实 Delphi的程序设计师早就考虑到了这一点,他们提供了 Zlib.pas和 Zlibconst.pas两个单元文件来解决数据压缩问题,实现了很高的数据压缩比率。这两个文件保存在 Delphi 5.0安装光盘上 /Info/Extras/Zlib目录下,此外,在 Info/Extras/Zlib/Obj目录中还保存了 Zlib.pas单元引用的 Obj文件。下面本文以压缩一个屏幕拷贝为例介绍如何使用这项功能。

  解决思路

  首先利用屏幕拷贝捕捉到当前整个屏幕的图像,然后在内存中保存为 BMP文件格式。压缩时,使用 TCompressionStream对象对原始图像进行压缩并且保存为自定义的文件格式;解压缩时,使用 TDecompressionStream对象对被压缩的图像进行解压缩,还原为 BMP格式的图像文件。

  具体实现

  新建一个项目文件,在主单元的接口部分引用 Zlib.pas,在主表单上放置两个按钮 Button1、 Button2,在它们的 onClick事件中写上相应的过程调用代码。

  部分程序源代码如下:

  unit Unit1;

  interface

  uses

   Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,StdCtrls, Zlib;

   type

    TForm1 = class(TForm)

    Button1: TButton;

    Button2: TButton;

    procedure Button1Click(Sender: TObject);

    procedure Button2Click(Sender: TObject);

   private

    { Private declarations }

   public

    { Public declarations }

   end;

   var

    Form1: TForm1;

    implementation

     {$ R* .DFM}

   1.捕捉全屏幕图像

    procedure GetScreen(var Bmp: TBitmap);

    var

     Dc: HDC;

     MyCanvas: TCanvas;

     MyRect: TRect;

    begin

     Dc := GetWindowDC(0);

     MyCanvas := TCanvas.Create;

    try

     MyCanvas.Handle := Dc;

     MyRect:=Rect(0, 0,Screen.Width, Screen.Height);

     file://图像为 24位真彩色,也可根据实际需要调整

     Bmp.PixelFormat := pf24bit;

     Bmp.Width := MyRect.Right;

     Bmp.Height := MyRect.Bottom;

     file://捕捉整个屏幕图像

     Bmp.Canvas.CopyRect(MyRect, MyCanvas, MyRect);

     finally

     MyCanvas.Handle := 0;

     MyCanvas.Free;

     ReleaseDC(0, Dc);

    end;

   end;

 2.压缩图像

   procedure CompressBitmap(var CompressedStream: TMemoryStream;const CompressionLevel: TCompressionLevel);

    var

     SourceStream: TCompressionStream;

     DestStream: TMemoryStream;

     Count: Integer;

    Begin

     file://获得图像流的原始尺寸

     Count := CompressedStream.Size;

     DestStream := TMemoryStream.Create;

     SourceStream:=TCompressionStream.Create

     (CompressionLevel, DestStream);

     Try

      file://SourceStream中保存着原始的图像流

      CompressedStream.SaveToStream(SourceStream);

      file://将原始图像流进行压缩, DestStream中保存着压缩后的图像流

      SourceStream.Free;

      CompressedStream.Clear;

      file://写入原始图像的尺寸

      CompressedStream.WriteBuffer(Count, SizeOf

      (Count));

      file://写入经过压缩的图像流

      CompressedStream.CopyFrom(DestStream, 0);

      finally

      DestStream.Free;

     end;

    end;


   3.还原被压缩图像

    procedure UnCompressBitmap(const CompressedStream: TFileStream; var Bmp: TBitmap);

      Buffer: PChar;

      Count: Integer;

     Begin

      file://从被压缩的图像流中读出原始图像的尺寸

      CompressedStream.ReadBuffer(Count, SizeOf(Count));

      file://根据图像尺寸大小为将要读入的原始图像流分配内存块

      GetMem(Buffer, Count);

      DestStream := TMemoryStream.Create;

      SourceStream := TDecompressionStream.Create(CompressedStream);

     Try

      file://将被压缩的图像流解压缩,然后存入 Buffer内存块中

      SourceStream.ReadBuffer(Buffer^, Count);

      file://将原始图像流保存至 DestStream流中

      DestStream.WriteBuffer(Buffer^, Count);

      DestStream.Position := 0;//复位流指针

      //从 DestStream流中载入原始图像流

      Bmp.LoadFromStream(DestStream);

      finally

      FreeMem(Buffer);

      DestStream.Free;

     end;

    end;

   4.压缩按钮 onClick事件

    procedure TForm1.Button1Click(Sender: TObject);

     var

      Bmp: TBitmap;

      CompressedStream: TMemoryStream;

     begin

      Bmp := TBitmap.Create;

      CompressedStream := TMemoryStream.Create;

     Try

       file://捕获当前整个屏幕 ,将图像保存至 Bmp对象中 GetScreen(Bmp);

      file://将 Bmp对象中的图像保存至内存流中

      Bmp.SaveToStream(CompressedStream);

      file://按缺省的压缩比例对原始图像流进行压缩

      CompressBitmap(CompressedStream, clDefault);

      file://将压缩之后的图像流保存为自定义格式的文件

      CompressedStream.SaveToFile(‘ C:/cj.dat’ );

      finally

      Bmp.Free;

      CompressedStream.Free;

     end;

    end;

   5.解压缩按钮 onClick事件

     procedure TForm1.Button2Click(Sender: TObject);

      var

       CompressedStream: TFileStream;

       Bmp: TBitmap;

      begin

       Bmp := TBitmap.Create;

       file://以文件流的只读方式打开自定义的压缩格式文件

       CompressedStream := TFileStream.Create(‘ C:/cj.dat’ , fmOpenRead);

      Try

       file://将被压缩的图像流进行解压缩

       UnCompressBitmap(CompressedStream, Bmp);

       file://将原始图像流还原为指定的 BMP文件

       Bmp.SaveToFile(‘ C:/cj.bmp’ );

       finally

       Bmp.Free;

       CompressedStream.Free;

      end;

      end;

  此外 TCompressionStream对象还提供了 CompressionRate属性,该属性用于描述对原始数据进行压缩后的压缩比率,而 OnProgress事件在压缩与解压缩过程中都会被触发,开发人员可以在该事件中编写用于显示进度的代码。

  以上代码在 Delphi 5.0中调试运行通过。  


来自:阿曼, 时间:2004-2-12 19:20:23, ID:2450401
[?谢谢。我用的是delphi 7, 以下是我的程序,但是在恢复的时候出错了。
我看了一下count的数目,在写进去和读出来的都是准确的,但是就是在恢复的时候错了。(在代码中标出了出错的地方)
我的代码好像没有错,为什么会出错呢?


procedure TForm1.CompressBitmap(var CompressedStream: TMemoryStream;const CompressionLevel: TCompressionLevel);
var SourceStream: TCompressionStream;
    DestStream: TMemoryStream;
    Count: Integer;
Begin
  //获得图像流的原始尺寸
  Count := CompressedStream.Size;
  DestStream := TMemoryStream.Create;
  SourceStream:=TCompressionStream.Create(CompressionLevel, DestStream);
  Try
    //SourceStream中保存着原始的图像流
    CompressedStream.SaveToStream(SourceStream);
    //将原始图像流进行压缩, DestStream中保存着压缩后的图像流
    SourceStream.Free;
    CompressedStream.Clear;
    //写入原始图像的尺寸
    CompressedStream.WriteBuffer(Count, SizeOf(Count));
    //写入经过压缩的图像流
    CompressedStream.CopyFrom(DestStream, 0);
  finally
    DestStream.Free;
  end;
end;

procedure TForm1.UnCompressBitmap(const CompressedStream: TFileStream; var Bmp: TBitmap);
var DestStream: TMemoryStream;
    SourceStream: TDecompressionStream;
    Buffer: PChar;
    Count: Integer;
Begin
  //从被压缩的图像流中读出原始图像的尺寸
  CompressedStream.ReadBuffer(Count, SizeOf(Count));
  //根据图像尺寸大小为将要读入的原始图像流分配内存块
  GetMem(Buffer, Count);
  DestStream := TMemoryStream.Create;
  SourceStream := TDecompressionStream.Create(CompressedStream);
  Try
    //将被压缩的图像流解压缩,然后存入 Buffer内存块中
    SourceStream.ReadBuffer(Buffer^, Count);
    //将原始图像流保存至 DestStream流中
    DestStream.WriteBuffer(Buffer^, Count);
    DestStream.Position := 0;//复位流指针
    //从 DestStream流中载入原始图像流
[red]
[?]//*********************出错点******************
[?]    Bmp.LoadFromStream(DestStream);       //  *
[?]//*********************************************
[/red]
  finally
    FreeMem(Buffer);
    DestStream.Free;
  end;
end;

procedure TForm1.GetScreen(var Bmp: TBitmap);
var Dc: HDC;
    MyCanvas: TCanvas;
    MyRect: TRect;
begin
  Dc := GetWindowDC(0);
  MyCanvas := TCanvas.Create;
  try
    MyCanvas.Handle := Dc;
    MyRect:=Rect(0, 0,Screen.Width, Screen.Height);
    //图像为 24位真彩色,也可根据实际需要调整
    Bmp.PixelFormat := pf24bit;
    Bmp.Width := MyRect.Right;
    Bmp.Height := MyRect.Bottom;
    //捕捉整个屏幕图像
    Bmp.Canvas.CopyRect(MyRect, MyCanvas, MyRect);
  finally
    MyCanvas.Handle := 0;
    MyCanvas.Free;
    ReleaseDC(0, Dc);
  end;
end;

procedure TForm1.BitBtn1Click(Sender: TObject);
var CompressedStream: TMemoryStream;
    Bmp: TBitmap;
begin
  CompressedStream := TMemoryStream.Create;
  Bmp := TBitmap.Create;
  Try
    //捕获当前整个屏幕 ,将图像保存至 Bmp对象中 GetScreen(Bmp);
    //将 Bmp对象中的图像保存至内存流中
    // bmp := image1.Picture.Bitmap;
    GetScreen(bmp);
    bmp.SaveToStream(CompressedStream);
    //按缺省的压缩比例对原始图像流进行压缩
    CompressBitmap(CompressedStream, clDefault);
    //将压缩之后的图像流保存为自定义格式的文件
    CompressedStream.SaveToFile('C:/cj.dat');
  finally
    bmp.Free;
    CompressedStream.Free;
  end;
end;

procedure TForm1.BitBtn2Click(Sender: TObject);
var Bmp: TBitmap;
    CompressedStream: TFileStream;
begin
  //以文件流的只读方式打开自定义的压缩格式文件
  CompressedStream := TFileStream.Create('C:/cj.dat', fmOpenRead);
  Try
    //将被压缩的图像流进行解压缩
    UnCompressBitmap(CompressedStream, bmp);
    //将原始图像流还原为指定的 BMP文件
    Bmp.SaveToFile('C:/cj.bmp');
  finally
    bmp.Free;
    CompressedStream.Free;
  end;
end;  

 

 

procedure TFORM.SAVE_PICTURE(Sender: TObject);
var
  tempStream:TMemoryStream;
  JpgPic:TJpegImage;
  S:String;
begin
 try
  JpgPic:=TJpegImage.Create;
  tempStream:=TMemoryStream.Create;
  tempStream.Clear;
  CDS.Edit;
  if openpicturedialog1.Execute then
   begin
    JpgPic.LoadFromFile(OpenPictureDialog1.FileName);
    image1.Picture.bitmap.assign(JpgPic);
    JpgPic.SaveToStream(tempStream);
    TBlobField(CDS.FieldByName('picture')).LoadFromStream(tempStream) ;
    CDS.ApplyUpdates(0);
   end;
 finally
  JpgPic.Free;
  tempStream.Free;
 end;
end;

取 :
procedure TFORM.show_picture;
var
  MyJpeg:TJpegImage;
  MyStm:TMemoryStream;
begin
 if CDS.FieldByName('picture').IsNull then
  begin
   image1.Picture:=nil;
   exit;
  end;
 try
   MyJpeg:=TJpegImage.Create;
   MyStm:=TMemoryStream.Create;
   MyStm.Clear;
   TBlobField(CDS.FieldByName('picture')).SaveToStream(MyStm);
   MyStm.Position:=0;
   MyJpeg.LoadFromStream(MyStm);
   image1.Picture.BitMap.Assign(MyJpeg);
 finally
   MyJpeg.Free;
   MyStm.Free;
 end;  

 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值