delphi 中的流

news/2024/7/4 23:07:31 标签: delphi, button, function, exe, 加密, stream

  什么是流?流,简单来说就是建立在面向对象基础上的一种抽象的处理数据
的工具。在流中,定义了一些处理数据的基本操作,如读取数据,写入数据等,
程序员是对流进行所有操作的,而不用关心流的另一头数据的真正流向。流不
但可以处理文件,还可以处理动态内存、网络数据等多种数据形式。如果你对
流的操作非常熟练,在程序中利用流的方便性,写起程序会大大提高效率的。
  下面,笔者通过四个实例:EXE文件加密器、电子贺卡、自制OICQ和网络屏幕
传输来说明Delphi编程中“流”的利用。这些例子中的一些技巧曾经是很多软
件的秘密而不公开的,现在大家可以无偿的直接引用其中的代码了。
  “万丈高楼平地起”,在分析实例之前,我们先来了解一下流的基本概念和
函数,只有在理解了这些基本的东西后我们才能进行下一步。请务必认真领会
这些基本方法。当然,如果你对它们已经很熟悉了,则可以跳过这一步。

一、Delphi中流的基本概念及函数声明
  在Delphi中,所有流对象的基类为TStream类,其中定义了所有流的共同属性
和方法。
  TStream类中定义的属性介绍如下:
  1、Size:此属性以字节返回流中数据大小。
  2、Position:此属性控制流中存取指针的位置。
  Tstream中定义的虚方法有四个:
  1、Read:此方法实现将数据从流中读出。函数原形为:
   Function Read(var Buffer;Count:Longint):Longint;virtual;abstract;
   参数Buffer为数据读出时放置的缓冲区,Count为需要读出的数据的字节数,
该方法返回值为实际读出的字节数,它可以小于或等于Count中指定的值。
  2、Write:此方法实现将数据写入流中。函数原形为:
   Function Write(var Buffer;Count:Longint):Longint;virtual;abstract;
   参数Buffer为将要写入流中的数据的缓冲区,Count为数据的长度字节数,
该方法返回值为实际写入流中的字节数。
  3、Seek:此方法实现流中读取指针的移动。函数原形为:
  Function Seek(Offset:Longint;Origint:Word):Longint;virtual;abstract;
  参数Offset为偏移字节数,参数Origint指出Offset的实际意义,其可能的取值
如下:
  soFromBeginning:Offset为移动后指针距离数据开始的位置。此时Offset必须
                  大于或者等于零。
  soFromCurrent:Offset为移动后指针与当前指针的相对位置。
  soFromEnd:Offset为移动后指针距离数据开始的位置。此时Offset必须
                  小于或者等于零。
该方法返回值为移动后指针的位置。
  4、Setsize:此方法实现改变数据的大小。函数原形为:
   Function Setsize(NewSize:Longint);virtual;
另外,TStream类中还定义了几个静态方法:
  1、ReadBuffer:此方法的作用是从流中当前位置读取数据。函数原形为:
   Procedure ReadBuffer(var Buffer;Count:Longint);
   参数的定义跟上面的Read相同。注意:当读取的数据字节数与需要读取的字节
数不相同时,将产生EReadError异常。
  2、WriteBuffer:此方法的作用是在当前位置向流写入数据。函数原形为:
   Procedure WriteBuffer(var Buffer;Count:Longint);
   参数的定义跟上面的Write相同。注意:当写入的数据字节数与需要写入的字节
数不相同时,将产生EWriteError异常。
  3、CopyFrom:此方法的作用是从其它流中拷贝数据流。函数原形为:
   Function CopyFrom(Source:TStream;Count:Longint):Longint;
   参数Source为提供数据的流,Count为拷贝的数据字节数。当Count大于0时,
CopyFrom从Source参数的当前位置拷贝Count个字节的数据;当Count等于0时,
CopyFrom设置Source参数的Position属性为0,然后拷贝Source的所有数据;
  TStream还有其它派生类,其中最常用的是TFileStream类。使用TFileStream
类来存取文件,首先要建立一个实例。声明如下:
constructor Create(const Filename:string;Mode:Word);
Filename为文件名(包括路径),参数Mode为打开文件的方式,它包括文件的打
开模式和共享模式,其可能的取值和意义如下:

打开模式:
fmCreate       :用指定的文件名建立文件,如果文件已经存在则打开它。
fmOpenRead     :以只读方式打开指定文件
fmOpenWrite    :以只写方式打开指定文件
fmOpenReadWrite:以写写方式打开指定文件
共享模式:
fmShareCompat   :共享模式与FCBs兼容
fmShareExclusive:不允许别的程序以任何方式打开该文件
fmShareDenyWrite:不允许别的程序以写方式打开该文件
fmShareDenyRead :不允许别的程序以读方式打开该文件
fmShareDenyNone :别的程序可以以任何方式打开该文件

   TStream还有一个派生类TMemoryStream,实际应用中用的次数也非
常频繁。它叫内存流,就是说在内存中建立一个流对象。它的基本方法和函数跟
上面是一样的。
   好了,有了上面的基础后,我们就可以开始我们的编程之行了。
-----------------------------------------------------------------------
二、实际应用之一:利用流制作EXE文件加密器、捆绑、自解压文件及安装程序

我们先来说一下如何制作一个EXE文件加密器吧。
EXE文件加密器的原理:建立两个文件,一个用来添加资源到另外一个EXE文件
里面,称为添加程序。另外一个被添加的EXE文件称为头文件。该程序的功能是
把添加到自己里面的文件读出来。
  Windows下的EXE文件结构比较复杂,有的程序还有校验和,当发现自己被改变
后会认为自己被病毒感染而拒绝执行。所以我们把文件添加到自己的程序里面,
这样就不会改变原来的文件结构了。我们先写一个添加函数,该函数的功能是把
一个文件当作一个流添加到另外一个文件的尾部。函数如下:

Function Cjt_AddtoFile(SourceFile,TargetFile:string):Boolean;
var
Target,Source:TFileStream;
MyFileSize:integer;
begin
try
Source:=TFileStream.Create(SourceFile,fmOpenRead or fmShareExclusive);
Target:=TFileStream.Create(TargetFile,fmOpenWrite or fmShareExclusive);
try
Target.Seek(0,soFromEnd);//往尾部添加资源
Target.CopyFrom(Source,0);
MyFileSize:=Source.Size+Sizeof(MyFileSize);//计算资源大小,并写入辅程尾部
Target.WriteBuffer(MyFileSize,sizeof(MyFileSize));
finally
Target.Free;
Source.Free;
end;
except
Result:=False;
Exit;
end;
Result:=True;
end;
   有了上面的基础,我们应该很容易看得懂这个函数。其中参数SourceFile是
要添加的文件,参数TargetFile是被添加到的目标文件。比如说把a.exe添加到
b.exe里面可以:Cjt_AddtoFile('a.exe',b.exe');如果添加成功就返回True否则
返回假。
  根据上面的函数我们可以写出相反的读出函数:
Function Cjt_LoadFromFile(SourceFile,TargetFile :string):Boolean;
var
Source:TFileStream;
Target:TMemoryStream;
MyFileSize:integer;
begin
try
Target:=TMemoryStream.Create;
Source:=TFileStream.Create(SourceFile,fmOpenRead or fmShareDenyNone);
try
Source.Seek(-sizeof(MyFileSize),soFromEnd);
Source.ReadBuffer(MyFileSize,sizeof(MyFileSize));//读出资源大小
Source.Seek(-MyFileSize,soFromEnd);//定位到资源位置
Target.CopyFrom(Source,MyFileSize-sizeof(MyFileSize));//取出资源
Target.SaveToFile(TargetFile);//存放到文件
finally
Target.Free;
Source.Free;
end;
except
Result:=false;
Exit;
end;
Result:=true;
end;
   其中参数SourceFile是已经添加了文件的文件名称,参数TargetFile是取出文
件后保存的目标文件名。比如说Cjt_LoadFromFile('b.exe','a.txt');在b.exe
取出文件保存为a.txt。如果取出成功就返回True否则返回假。
   打开Delphi,新建一个工程,在窗口上放上一个Edit控件Edit1和两个Button:
Button1和Button2。Button的Caption属性分别设置为“确定”和“取消”。在
Button1的Click事件中写代码:
var S:string;
begin
  S:=ChangeFileExt(Application.ExeName,'.Cjt');
  if Edit1.Text='790617' then
     begin
       Cjt_LoadFromFile(Application.ExeName,S);
       {取出文件保存在当前路径下并命名"原文件.Cjt"}
       Winexec(pchar(S),SW_Show);{运行"原文件.Cjt"}
       Application.Terminate;{退出程序}
     end
    else
      Application.MessageBox('密码不对,请重新输入!','密码错误',MB_ICONERROR+MB_OK);
   编译这个程序,并把EXE文件改名为head.exe。新建一个文本文件head.rc,
内容为: head exefile head.exe,然后把它们拷贝到Delphi的BIN目录下,执行
Dos命令Brcc32.exe head.rc,将产生一个head.res的文件,这个文件就是我们要
的资源文件,先留着。
  我们的头文件已经建立了,下面我们来建立添加程序。
  新建一个工程,放上以下控件:一个Edit,一个Opendialog,两个Button1的
Caption属性分别设置为"选择文件"和"加密"。
  在源程序中添加一句:{$R head.res}并把head.res文件拷贝到程序当前目录下。
这样一来就把刚才的head.exe跟程序一起编译了。
  在Button1的Cilck事件里面写下代码:
if OpenDialog1.Execute then  Edit1.Text:=OpenDialog1.FileName;
  在Button2的Cilck事件里面写下代码:
var S:String;
begin
S:=ExtractFilePath(Edit1.Text);
if ExtractRes('exefile','head',S+'head.exe') then
   if Cjt_AddtoFile(Edit1.Text,S+'head.exe') then
      if DeleteFile(Edit1.Text) then
        if RenameFile(S+'head.exe',Edit1.Text) then
          Application.MessageBox('文件加密成功!','信息',MB_ICONINFORMATION+MB_OK)
   else
     begin
     if FileExists(S+'head.exe') then DeleteFile(S+'head.exe');
     Application.MessageBox('文件加密失败!','信息',MB_ICONINFORMATION+MB_OK)
     end;
end;
  其中ExtractRes为自定义函数,它的作用是把head.exe从资源文件中取出来。
Function ExtractRes(ResType, ResName, ResNewName : String):boolean;
var
Res : TResourceStream;
begin
try
Res := TResourceStream.Create(Hinstance, Resname, Pchar(ResType));
try
Res.SavetoFile(ResNewName);
Result:=true;
finally
Res.Free;
end;
except
Result:=false;
end;
end;
  注意:我们上面的函数只不过是简单的把一个文件添加到另一个文件的尾部。
实际应用中可以改成可以添加多个文件,只要根据实际大小和个数定义好偏移
地址就可以了。比如说文件捆绑机就是把两个或者多个程序添加到一个头文件
里面。那些自解压程序和安装程序的原理也是一样的,不过多了压缩而已。
比如说我们可以引用一个LAH单元,把流压缩后再添加,这样文件就会变的很小。
读出来时先解压就可以了。
  另外,文中EXE加密器的例子还有很多不完善的地方,比如说密码固定为
"790617",取出EXE运行后应该等它运行完毕后删除等等,读者可以自行修改。 

-----------------------------------------------------------------------------------------------------------------

TFileStream对象

  TFileStream对象是在磁盘文件上存储数据的Stream对象。TFileStream是从THandleStream继承下来的,它和THandleStream一样都是实现文件的存取操作。不同之处在于THandleStream用句柄访问文件,而TFileStream用文件名访问文件。实际上TFileStream是THandleStream上的一层包装,其内核是THandleStream的属性和方法。

  TFileStream中没有增加新的属性和方法。它只是覆盖了的构造方法Create和析构方法Destory。在Create方法中带两个参数FileName和Mode。FileName描述要创建或打开的文件名,而Mode描述文件模式如fmCreate、fmOpenRead和fmOpenWrite等。Create方法首先使用FileCreate或FileOpen函数创建或打开名为FileName的文件,再将得到的文件句柄赋给FHandle。TFileStream的文件读写操作都是由从THandleStream继承的Read

var

Stream: TStream;

begin

Stream := TFileStream.Create(FileName, fmCreate);

try

SaveToStream(Stream);

finally

Stream.Free;

end;

end;

  在Delphi 的许多对象的SaveToStream 和SaveToFile、LoadFromStream和LoadFromFile方法的实现都有类似的嵌套结构。


http://www.niftyadmin.cn/n/1415922.html

相关文章

JQuery+CSS实现点击列状菜单

JQueryCSS实现点击列状菜单开发环境实现效果代码实现开发环境 Jquery-3.3.1,CSS 实现效果 鼠标滑过和点击时菜单改变底色和字体大小 正常状态 鼠标滑过 鼠标点击 2. 点击主菜单时,对应子菜单切换显示状态,其他非子菜单隐藏 原始状态 …

多线程编程(18) - 再从一个小例子出发

为什么80%的码农都做不了架构师?>>> 前面的例子都是让若干线程做同样的事情, 下面这个例子中的三个线程将分别在三个画板上随机画不同颜色的椭圆. 接下来的很多事情我想要基于这个例子来做. 本例效果图: 代码文件: unit Unit1;interfaceusesWindows, Me…

满江红

满江红 写怀 怒发冲冠,凭栏处、潇潇雨歇。 抬望眼,仰天长啸,壮怀激烈。 三十功名尘与土,八千里路云和月。 莫等闲、白了少年头,空悲切。 靖康耻,犹未雪。 臣子恨,何时灭。 驾长车&…

php 从结果集中取出第一个,php – 如何将查询的第一个结果与其他结果分开?...

我想将第一个PDO结果数组与其他数组分开.这是我的尝试:$iterator 1;while($results $stmt->fetch(PDO::FETCH_ASSOC) and $iterator 1){/* I need just first row here */print_r($results);$iterator;}while($results $stmt->fetch(PDO::FETCH_ASSOC)){/…

IntelliJ IDEA 2017 上传本地项目至码云

码云免费还挺好用,和Git类似。首先需要注册一个码云。这个就不介绍了。 点击加号,新建一个项目。 填写项目名称。 复制一下你的项目地址。 如图勾选。将项目至于git管理。 首先commit。快捷按钮在右上角。 接着将刚才commit的代码push到码云上。 填写刚才…

JQuery+CSS实现列表清单

JQueryCSS实现列表清单开发环境实现效果代码实现开发环境 Jquery-3.3.1&#xff0c;CSS 实现效果 表格内容显示&#xff0c;分页器样式 代码实现 HTML部分 <!DOCTYPE html> <head><title>浮动菜单</title><link rel"stylesheet" t…

虚拟键码对照表

为什么80%的码农都做不了架构师&#xff1f;>>> 虚拟键码 对应值 对应键VK_LBUTTON1鼠标左键VK_RBUTTON2鼠标右键VK_CANCEL3CancelVK_MBUTTON4鼠标中键VK_XBUTTON15VK_XBUTTON26VK_BACK8BackspaceVK_TAB9TabVK_CLEAR12ClearVK_RETURN13EnterVK_SHIFT16ShiftVK_CONT…

think php mvc,thinkphp中mvc代表什么

ThinkPHP基于MVC(Model-View-Controller&#xff0c;模型-视图-控制器)模式&#xff0c;并且均支持多层(multi-Layer)设计。模型(Model)层默认的模型层由Model类构成&#xff0c;但是随着项目的增大和业务体系的复杂化&#xff0c;单一的模型层很难解决要求&#xff0c;ThinkPH…