delphi内存映射 与 映射数据获取
一.原理
通过使用“内存映射文件”,实现内存共享
二.主要操作
共享内存结构:
PShareMem = ^TShareMem;
TShareMem = Record
id:string[10];
name:string[20];
age:Integer;
end; // 一定要注意 固定长度
基本变量:
shareMemName:string; //共享内存名
fileHandle : THandle;//内存映射文件句柄
pUserInfoShareMem : PShareMem;//指向共享内存的指针
a)写入程序
1)创建“内存映射文件”
begin
//创建“内存映射文件”
fileHandle:=CreateFileMapping($FFFFFFFF, nil, PAGE_READWRITE, 0, SizeOf(TShareMem), PChar(shareMemName));
if fileHandle <> 0 then
begin
Self.Memo1.Lines.Add('已成功创建内存映射文件!');
end;
end;
2)建立映射关系
//将“内存映射文件”与“应用程序地址空间”建立映射关系
pUserInfoShareMem:=MapViewOfFile(fileHandle,FILE_MAP_ALL_ACCESS,0,0,sizeof(TShareMem));
if pUserInfoShareMem <> nil then
begin
Self.Memo1.Lines.Add('已成功建立映射关系!');
end;
3)写入信息
pUserInfoShareMem.id:='8888';
pUserInfoShareMem.name:='Terry';
pUserInfoShareMem.age:=25;
Self.Memo1.Lines.Add('已向共享内存中写入用户信息!');
4)解除映射关系
//解除“内存映射文件”与“应用程序地址空间”的映射关系
if pUserInfoShareMem<> nil then
UnmapViewOfFile(pUserInfoShareMem);
Self.Memo1.Lines.Add('已成功解除映射关系!');
5)关闭“内存映射文件”
//关闭内存映射文件
if fileHandle<> 0 then
CloseHandle(fileHandle);
Self.Memo1.Lines.Add('已成功关闭内存映射文件!');
b)读取程序
1)打开“内存映射文件”
fileHandle:=OpenFileMapping(FILE_MAP_ALL_ACCESS,false,pchar(shareMemName));
if self.FileHandle <> 0 then
begin
Self.Memo1.Lines.Add('已成功打开内存映射文件!')
end;
2)建立映射关系
pUserInfoShareMem:= MapViewOfFile(self.FileHandle,windows.FILE_MAP_ALL_ACCESS,0,0,sizeof(TShareMem));
if pUserInfoShareMem <> nil then
begin
Self.Memo1.Lines.Add('已成功建立映射关系!');
end;
3)读取信息
if pUserInfoShareMem <> nil then
begin
userInfoStr:='共享内存中获取的用户信息如下:'+#13#10;
userInfoStr:=userInfoStr+'用户Id号:'+pUserInfoShareMem.id+#13#10;
userInfoStr:=userInfoStr+'用户姓名:'+pUserInfoShareMem.name+#13#10;
userInfoStr:=userInfoStr+'用户年龄:'+IntToStr(pUserInfoShareMem.age);
Self.Memo1.Lines.Add(userInfoStr);
end;
4)解除映射关系
if pUserInfoShareMem<> nil then
UnmapViewOfFile(pUserInfoShareMem);
Self.Memo1.Lines.Add('已成功解除映射关系!');
5)关闭“内存映射文件”
if fileHandle<> 0 then
CloseHandle(fileHandle);
Self.Memo1.Lines.Add('已成功关闭内存映射文件!');
====================================================
以下 转自 http://blog.csdn.net/bdmh/article/details/6369250
procedure TGetDataThread.DoGetData;
var
FFile_Handle: THandle;
FFile_Map: THandle;
list: TStringList;
p: PChar;
i, interval: Integer; begin
try
totallen := ;
offset := ;
tstream := TMemoryStream.Create;
stream := TMemoryStream.Create;
list := TStringList.Create;
// 获取系统信息
GetSystemInfo(sysinfo);
// 页面分配粒度大小
blocksize := sysinfo.dwAllocationGranularity;
// 打开文件
FFile_Handle := CreateFile(PChar(FSourceFileName), GENERIC_READ,
FILE_SHARE_READ, nil, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, );
if FFile_Handle = INVALID_HANDLE_VALUE then
Exit;
// 获取文件尺寸
filesize := GetFileSize(FFile_Handle, nil);
// 创建映射
FFile_Map := CreateFileMapping(FFile_Handle, nil, PAGE_READONLY, , , nil);
if FFile_Map = then
Exit;
// 此处我们已10倍blocksize为一个数据块来映射,如果文件尺寸小于10倍blocksize,则直接映射整个文件长度
if filesize div blocksize > then
readlen := * blocksize
else
readlen := filesize;
for i := to FInfoList.Count - do
begin
list.Delimiter := ':';
list.DelimitedText := FInfoList.Strings[i];
// 取得长度,我这里做了解析,因为我存储的信息为 a:b:c 这种类型,所以以:号分隔
len := StrToInt(list.Strings[]);
interval := StrToInt(list.Strings[]);
if (i = ) or (totallen + len >= readlen) then
begin
// 如果已读取的长度加上即将要读取的长度大于 10倍blocksize,那么我们要保留之前映射末尾的内容,以便和新映射的内容合并
if i > then
begin
offset := offset + readlen;
// 写入临时流
tstream.Write(p^, readlen - totallen);
tstream.Position := ;
end;
// 如果未读取的数据长度已经不够一个分配粒度,那么就直接映射剩下的长度
if filesize - offset < blocksize then
readlen := filesize - offset;
// 映射,p是指向映射区域的指针
// 注意这里第三个参数,一直设为0,这个值要根据实际情况设置
p := PChar(MapViewOfFile(FFile_Map, FILE_MAP_READ, , offset, readlen));
end;
// 如果临时流中有数据,需要合并
if tstream.Size > then
begin
// 把临时流数据copy过来
stream.CopyFrom(tstream, tstream.Size);
// 然后在末尾写入新数据,合并完成
stream.Write(p^, len - tstream.Size);
totallen := len - tstream.Size;
// 移动指针的位置,指向下一个数据的开始
Inc(p, len - tstream.Size);
tstream.Clear;
end
else
begin
stream.Write(p^, len);
totallen := totallen + len;
Inc(p, len);
end;
stream.Position := ;
// 将流保存成文件
stream.SaveToFile(IntToStr(i) + '.txt');
stream.Clear;
end;
finally
stream.Free;
tstream.Free;
CloseHandle(FFile_Handle);
CloseHandle(FFile_Map);
end;
end;
delphi内存映射 与 映射数据获取的更多相关文章
- delphi 内存映射
使用内存映射文件读写大文件 使用内存映射文件读写大文件 文件操作是应用程序最为基本的功能之一,Win32 API和MFC均提供有支持文件处理的函数和类.一般来说,这些函数可以满足大多数场合的要求,但是 ...
- Delphi内存操作API函数(备查,并一一学习)
Delphi内存操作API函数System.IsMemoryManagerSet;System.Move;System.New;System.ReallocMem;System.ReallocMemo ...
- Delphi 内存与指针
源:Delphi 内存与指针 Delphi 的内存操作函数(1): 给字符指针分配内存 Delphi 的内存操作函数(2): 给数组指针分配内存 Delphi 的内存操作函数(3): 给结构体指针分配 ...
- Delphi 内存分配 StrAlloc New(转)
源:Delphi 内存分配 StrAlloc New 引自:http://anony3721.blog.163.com/blog/static/5119742010824934164/ 给字符指针 ...
- 【学习笔记】Hibernate 一对一关联映射 组件映射 二级缓存 集合缓存
啊讲道理放假这十天不到啊 感觉生活中充满了绝望 这就又开学了 好吧好吧继续学习笔记?还是什么的 一对一关联映射 这次我们仍然准备了两个表 一个是用户表Users 一个是档案表Resume 他们的关系是 ...
- JPA(六):映射关联关系------映射单向一对多的关联关系
映射单向一对多的关联关系 新建项目项目请参考<JPA(二):HellWord工程>,基于上一章讲解的<JPA(五):映射关联关系------映射单向多对一的关联关系>中的例子进 ...
- 关于Delphi内存表的使用说明
关于Delphi内存表的使用说明: 1.建立临时表 数据输入是开发数据库程序的必然环节.在Client/Server结构中,客户端可能要输入一批数据后,再向服务器的后台数据库提交,这就需要在本地(客 ...
- [二]Java虚拟机 jvm内存结构 运行时数据内存 class文件与jvm内存结构的映射 jvm数据类型 虚拟机栈 方法区 堆 含义
前言简介 class文件是源代码经过编译后的一种平台中立的格式 里面包含了虚拟机运行所需要的所有信息,相当于 JVM的机器语言 JVM全称是Java Virtual Machine ,既然是虚拟机, ...
- linux I/O 内存分配和映射
I/O 内存区必须在使用前分配. 分配内存区的接口是( 在 <linux/ioport.h> 定义): struct resource *request_mem_region(unsign ...
随机推荐
- vue安装流程
一.环境搭建 vue推荐开发环境: Node.js: javascript运行环境(runtime),不同系统直接运行各种编程语言 npm: Nodejs下的包管理器.由于国内使用npm会很慢,这 ...
- SQL注入之Sqli-labs系列第十五关和第十六关(基于POST的时间盲注)
开始挑战第十五关(Blind- Boolian Based- String)和 第十六关(Blind- Time Based- Double quotes- String) 访问地址,输入报错语句 ' ...
- linux 将一个文件分解成多个不同名文件
1.通过c直接实现 #include <stdio.h> #include <stdlib.h> #include <sys/types.h> #include & ...
- JAVA小程序-----买衣服
import java.util.Scanner; //引用扫描器 public class TestDemo1 { public static void main(String [] args){ ...
- mysql str_to_date 字符串 转日期时间
SELECT STR_TO_DATE('2018-05-05 14:00:00.5555','%Y-%m-%d %H:%i:%s') from DUAL;
- 当BeanUtils遇到泛型
前言: BeanUtils(spring版/apache版)工具极大方便了java developer, 尤其在写业务代码中, 各种域模型DO, BO, VO等对象之间的复制. 但使用BeanUtil ...
- mysql手动设置数据表的自增值
设置表tablename的自增值从1开始自增值 alter table tablename auto_increment=1;
- [LeetCode&Python] Problem 13. Roman to Integer
Roman numerals are represented by seven different symbols: I, V, X, L, C, D and M. Symbol Value I 1 ...
- 初学html的单词笔记
font-size: 文字大小color: 顏色solid: 边框线text-align: 間距center: 文字放在中間<head> 网页头部<title> 网页标题< ...
- 大型网站系统与Java中间件实践读书笔记
转载:http://blog.csdn.net/ioscodelover/article/details/45047869 1.分布式系统相对集中式而言,是指多台计算机互相通过消息通信进行协作而对外提 ...