在Dephi中提供了一个抽象的数据类型TStream来支持对流式数据的操作。这些数据通常来自文件、数据库、内存对象、OLE对象等,TStream提供了统一、简洁的方法来进行数据的读写。在通常情况下,我们并不需要直接使用TStream类,对流式数据的读写封装在VCL控件的方法中。但是如果这些方法无法满足我们的要求,就需要自己手动控制数据的读写。

一、 TStream的常用的方法和属性:
1. function Read(var Buffer; Count: Longint): Longint; virtual; abstract

2. function Write(const Buffer; Count: Longint): Longint; virtual; abstract;

3. function Seek(Offset: Longint; Origin: Word): Longint; virtual; abstract;

4. property Position: Longint;

5. property Size: Longint

Read,Write,Seek都是纯虚函数,提供了数据读写和定位的抽象的方法。Read方法将数据从Stream中读到Buffer缓冲区中,Write则实现相反的操作,返回值表示实际读写数据的大小。Seek提供了在Stream中移动数据指针的方法。参数Origin可以取soFromBeginning,soFromCurrent,soFromEnd 三个值,Offset是偏移量,返回值是当前Stream数据指针的位置。

Position表示了数据指针在Stream中的位置。这个属性是可读写的,它实际上就是通过调用Seek方法实现的,所以实际使用时使用这个属性更为方便一些。Size属性表示当前Stream的大小,对于不同的Stream,有些时候是只读的。

二、 Stream数据的读写。
1. SaveToStream(Stream: TStream ); file://将类中的数据写到Stream的当前位置中

2. LoadFromStream(Stream: TStream); file://从当前位置读入Stream里的数据

实际使用时我们基本上只要使用上面两个函数就可以了。

三、 例子
TStream的继承树图如图1所示(略),实际使用时比较常用的是TFileStream,TMemoryStream,TblobStream,就以这三种流举一例说明具体用法。

创建一个窗体Form1,放置三个按钮btnRead,btnInvert,btnSave和一个文件打开对话框OpenDialog1以及数据控件DataSource1,Table1,test.

使用Dephi提供的Database Desktop创建一个表test,表里有一个字段域Image,数据库文件名存为test.db。在窗体上放置一个TDatabase控件dbTest,一个TTable控件Table1,一个DataSource控件DataSource1,一个TDBNavigator控件DBNavigator1。将dbTest与刚才Desktop创建的数据库相连,Table1的TableName属性设为test.db,DataSource1的DataSet属性设为Table1,DBNavigator1的DataSource属性设为DataSource1,VisibleButtons属性前四个设为TRUE。此外,将dbtest的Connected设为TRUE,Table1的Active属性设为TRUE,使得数据库一开始就处于打开状态。

事件代码编写如下:

1. btnRead的Click事件,这里演示了TFileStream的用法。

var
 MS: TFileStream;
begin
 if OpenDialog1.Execute then
 begin
MS:=TFileStream.Create
(OpenDialog1.FileName, fmOpenRead);
  Image1.Picture.Bitmap.LoadFromStream(MS);
  MS.Free;
 end;
end;
2. btnInvert的Click事件,这里演示了TMemoryStream的用法。其中使用了Invert函数,这是一个简单的将图象反色的函数(仅对真彩图象有效),它返回一个指向处理过的图象数据块的指针。

var
 M
S: TMemoryStream;
 pImage: pointer; 
begin
 MS:=TMemoryStream.create;
 Image1.Picture.Bitmap.SaveToStream(MS);
 MS.Position:=0;
 pImage:=Invert(MS.Memory, MS.size); 
file://Memory属性是指向实际内存块的指针
 MS.Write(pImage^,MS.size);
 MS.Position:=0;     
file://上一行代码使指针移到了Stream末尾,所以要复位
 Image1.Picture.Bitmap.LoadFromStream(MS);
 FreeMem(pImage); 
 MS.Free;
end;

Invert函数如下:
function TForm1.Invert
(pImage: pointer; size: Integer): pointer;
var
 pData, pMem: PChar;
 i: Integer;
begin
 pMem:=AllocMem(size);
 CopyMemory(pMem,pImage,size);
 pData:=pMem+54;
 for i:=0 to size-54-1 do
 begin
  pData^:=Char(not integer(pData^));
  pData:=pData+1;
 end;
 Result:=pMem;
end;
1. btnSave的Click事件,这里演示了TMemoryStream的另一种用法,将Stream中的数据写到数据库中去。

var
 MS: TMemoryStream;
begin
 MS:=TMemoryStream.create;
 Image1.Picture.Bitmap.SaveToStream(MS);
 MS.Position:=0;
 Table1.Append;  
file://在数据库中添加一条记录
 TBlobField(Table1.FieldbyName
('image')).LoadFromStream(MS);
 Table1.Post;    
file://将所作的更新写入数据库
end;
4. DBNavigator1的Click事件,这里演示了TBlobStream的用法,使用了和写入时不同的方法来读出数据库的图象数据。

var
 MS: TStream;
begin
 with Table1 do
 MS:=CreateBlobStream
(FieldbyName('image'),bmRead);
 Image1.Picture.Bitmap.
LoadFromStream(MS);
 MS.Free;
end; 
1、谈Delphi编程中“流”的利用

什么是流?流,简单来说就是建立在面向对象基础上的一种抽象的处理数据的工具。在流中,定义了一些处理数据的基本操作,如读取数据,写入数据等,程序员是对流进行所有操作的,而不用关心流的另一头数据的真正流向。流不但可以处理文件,还可以处理动态内存、网络数据等多种数据形式。如果你对流的操作非常熟练,在程序中利用流的方便性,写起程序会大大提高效率的。 
下面,笔者通过四个实例: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运行后应该等它运行完毕后删除等等,读者可以自行修改。

--------------------------------------------------------------------- 
三、实际应用之二:利用流制作可执行电子贺卡

我们经常看到一些电子贺卡之类的制作软件,可以让你自己选择图片,然后它会生成一个EXE可执行文件给你。打开贺卡时就会一边放音乐一边显示出图片来。现在学了流操作之后,我们也可以做一个了。 
添加图片过程我们可以直接用前面的Cjt_AddtoFile,而现在要做的是如何把图像读出并显示。我们用前面的Cjt_LoadFromFile先把图片读出来保存为文件再调入也是可以的,但是还有更简单的方法,就是直接把文件流读出来显示,有了流这个利器,一切都变的简单了。 
现在的图片比较流行的是BMP格式和JPG格式。我们现在就针对这两种图片写出读取并显示函数。

Function Cjt_BmpLoad(ImgBmp:TImage;SourceFile:String):Boolean; 
var 
Source:TFileStream; 
MyFileSize:integer; 
begin 
Source:=TFileStream.Create(SourceFile,fmOpenRead or fmShareDenyNone); 
try 
try 
Source.Seek(-sizeof(MyFileSize),soFromEnd); 
Source.ReadBuffer(MyFileSize,sizeof(MyFileSize));//读出资源 
Source.Seek(-MyFileSize,soFromEnd);//定位到资源开始位置 
ImgBmp.Picture.Bitmap.LoadFromStream(Source); 
finally 
Source.Free; 
end; 
except 
Result:=False; 
Exit; 
end; 
Result:=True; 
end; 
上面是读出BMP图片的,下面的是读出JPG图片的函数,因为要用到JPG单元,所以要在程序中添加一句:uses jpeg。

Function Cjt_JpgLoad(JpgImg:Timage;SourceFile:String):Boolean; 
var 
Source:TFileStream; 
MyFileSize:integer; 
Myjpg: TJpegImage; 
begin 
try 
Myjpg:= TJpegImage.Create; 
Source:=TFileStream.Create(SourceFile,fmOpenRead or fmShareDenyNone); 
try 
Source.Seek(-sizeof(MyFileSize),soFromEnd); 
Source.ReadBuffer(MyFileSize,sizeof(MyFileSize)); 
Source.Seek(-MyFileSize,soFromEnd); 
Myjpg.LoadFromStream(Source); 
JpgImg.Picture.Bitmap.Assign(Myjpg); 
finally 
Source.Free; 
Myjpg.free; 
end; 
except 
Result:=false; 
Exit; 
end; 
Result:=true; 
end; 
有了这两个函数,我们就可以制作读出程序了。下面我们以BMP图片为例: 
运行Delphi,新建一个工程,放上一个显示图像控件Image1。在窗口的Create事件中写上一句就可以了: 
Cjt_BmpLoad(Image1,Application.ExeName); 
这个就是头文件了,然后我们用前面的方法生成一个head.res资源文件。

在Dephi中使用TStream读写数据的技巧相关推荐

  1. linux sd卡中文件多时读写,数据存储与访问之——文件存储读写

    1.Android文件的操作模式 在java中要想对文件做读写操作,只需创建 文件,读写数据即可,Android却是不同,android基于Linux,在读写文件的时候,还需要加上文件的操作模式. 文 ...

  2. access 套用表格_Word表格编辑技巧:在Word中使用Access的数据-word技巧-电脑技巧收藏家...

    Word表格编辑技巧:在Word中使用Access的数据 利用下面的方法可以在原有Word文档中插入Access表或查询的内容: 1.在一篇Word文档中选定要插入表或查询的位置; 2.单击" ...

  3. 在unity2017中使用EXCEL读写数据

    1.导入ExcelSDK:链接:https://pan.baidu.com/s/1LiU6Kaxmc9j6kbFoE9QMFA?pwd=dr78 提取码:dr78 2.挂载脚本,并运行unity us ...

  4. python中json模块读写数据

    JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式,易于人阅读和编写 一.json字符串操作 python对象转成json字符串json.dumps()及js ...

  5. 在C#程序中三菱PLC读写数据

    首先添加引用 三菱PLCdll文件 1.导入 命名空间: using HslCommunication.Profinet.Melsec; 2.声明一个PLC对象: private MelsecMcNe ...

  6. MATLAB中文件的读写和数据的导入导出

    http://blog.163.com/tawney_daylily/blog/static/13614643620111117853933/ 在编写一个程序时,经常需要从外部读入数据,或者将程序运行 ...

  7. python应用中调用spark_在python中使用pyspark读写Hive数据操作

    1.读Hive表数据 pyspark读取hive数据非常简单,因为它有专门的接口来读取,完全不需要像hbase那样,需要做很多配置,pyspark提供的操作hive的接口,使得程序可以直接使用SQL语 ...

  8. android json mysql_Android通过json向MySQL中读写数据的方法详解【读取篇】

    本文实例讲述了Android通过json向MySQL中读取数据的方法.分享给大家供大家参考,具体如下: 首先 要定义几个解析json的方法parseJsonMulti,代码如下: private vo ...

  9. Java18-day09【字节缓冲流、字符流、编码表、字符串与字符流中的编码解码问题、字符流读写数据的方式、字符缓冲流、IO流小结】

    视频+资料(工程源码.笔记)[链接:https://pan.baidu.com/s/1MdFNUADVSFf-lVw3SJRvtg   提取码:zjxs] Java基础--学习笔记(零起点打开java ...

最新文章

  1. mywebsql java版_MyWebSQL|MySQL数据库管理软件(MyWebSQL)下载v3.7官方版 - 欧普软件下载...
  2. 十六进制转double
  3. 使用迁移学习在(选定)农业作物中的自动疾病分类
  4. 大数据分析如何助力企业发展业务
  5. 《集异璧》作者侯世达:王维、杨绛与机器翻译的本质
  6. js 生成二维码及打印
  7. MinGW-w64的安装及配置教程
  8. 钽电容的命名,贴片电解电容耐压,封装
  9. 在线识别图片中的字体的网站
  10. 怎么用计算机进行进制间的换算,如何实现16进制与其他进制之间的转换,教你使用16进制计算器...
  11. 28句最精辟有哲理的生活感悟说说,经典至极,总有一句说到你的心里
  12. ecu根据什么信号对点火提前角_刷ECU能让发动机秒变高功?工程师:你还太年轻...
  13. python3 下载.m3u8, 合并视频.ts 文件并合成为mp4格式的视频
  14. 【SQLServer】用SQL语句更改数据库名,表名,列名
  15. 关于mysql彻底卸除问题
  16. C++ 三阶魔方还原
  17. 大数据可视化,助力行业大数据应用
  18. grep -v grep使用说明
  19. java高级进阶知识整理
  20. 戴尔(DELL)台式机optiplex 7080安装centos经验

热门文章

  1. vue之vuex的五个属性
  2. php mysql宠物社交网站kf15168
  3. 鲁滨逊漂流记中的时代精神
  4. android模拟tcp接口转发,PC电脑和Android模拟器访问及模拟器之间tcp/udp通信
  5. 问界宣传海报增加“HUAWEI”前缀——“华为汽车”要来了?
  6. XMind之乱世三国
  7. 基于Python所写的超级画板设计
  8. CVE-2022-20261 Wordpress的插件SQL注入漏洞分析及修补
  9. 学计算机应用的上网课需要买电脑吗,大学生上网课有必要同时用笔记本电脑和ipad吗?...
  10. 深圳python编程培训