力为的技术博客

联系 聚合 管理
  154 Posts :: 1 Stories :: 561 Comments :: 0 Trackbacks

                                     如何通过FILE*操作内存文件

/*
问题概述:
  有这样一个第三方的库函数void WriteDataToFile(FILE* pFile),它是向一个文件写入数据。
  现在需要通过函数WriteDataToFile生成临时数据,但不能生成磁盘文件(需要在内存中操作)。
  然后通过某种方式读取内存中的数据。

流程设想:
1、
创建一个内存文件
获取该内存文件的FILE*
写入数据
2、
打开该内存文件
获取内存文件的FILE*
读取内存文件的内容
关闭
3、
关闭
*********************************************************************************
需要解决的问题:
1. 内存文件的HANDLE是否可以转换为FILE* (_open_osfhandle不可以)
2. 内存文件的操作需要调用PVOID pView = MapViewOfFile(...); 通过pView才能更改内存文件之内容。
   通过FILE* 能否通过某种方式操作pView的内容
3. 需要实现的函数(见下面的测试代码)
   FILE* GetFilePtrByMemFileHandle(HANDLE hMemFile)
   void UpdateMemFile(HANDLE hMemFile)
4. 如果该方案有问题,可以更改

**********************************************************************************
实现:
  1.为了方便测试使用全局的文件HANDLE
  2.模拟实现了WriteDataToFile函数
  3.已有资料http://community.csdn.net/Expert/TopicView.asp?id=4317083

 1*/
 2#define MF_NAME TEXT("MF_FILE_NAME")
 3#define MF_SIZE 4*1024
 4HANDLE g_hMemFile = NULL;
 5// 模拟库函数WriteDataToFile
 6void WriteDataToFile(FILE* pFile)
 7{
 8   ASSERT(pFile);
 9   fputs("Write by Lib Function!\n", pFile);
10}

11
12BOOL CreateMemFile()
13{
14g_hMemFile = CreateFileMapping(INVALID_HANDLE_VALUE, 
15NULL, PAGE_READWRITE, 0
16MF_SIZE, MF_NAME);
17if(g_hMemFile != NULL)
18{
19if(GetLastError() == ERROR_ALREADY_EXISTS)
20{
21TRACE("STL MapFile already exists.");
22CloseHandle(g_hMemFile);
23return FALSE;
24}

25return TRUE;
26}

27return FALSE;
28}

29
30FILE* GetFilePtrByMemFileHandle(HANDLE hMemFile)
31{
32//.
33//具体的实现
34
35//.
36return NULL;
37}

38void UpdateMemFile(HANDLE hMemFile)
39{
40//..
41//需要实现
42}

43
44void Test()
45{
46// 1 写内存文件
47if(CreateMemFile() == FALSE)
48return;
49
50     FILE* pFile = GetFilePtrByMemFileHandle(g_hMemFile);
51 if(NULL == pFile)
52 {
53 CloseHandle(g_hMemFile);
54 return;
55 }

56
57 WriteDataToFile(pFile);
58 //更新内存文件
59 UpdateMemFile(g_hMemFile);
60
61// 2 读内存文件
62 HANDLE hFile = OpenFileMapping(FILE_MAP_READ | FILE_MAP_WRITE, FALSE, MF_NAME);
63 if(hFile != NULL)
64 {
65 FILE* pReadFile = GetFilePtrByMemFileHandle(hFile);
66 if(pReadFile)
67 {
68 char strBuffer[BUFSIZ];
69            int nPtCount = 0;
70            while(fgets(strBuffer, BUFSIZ, pReadFile) != NULL)
71{
72TRACE(strBuffer);
73}

74 }

75 CloseHandle(hFile);
76 }

77// 3 关闭 
78 CloseHandle(g_hMemFile);
79}

80
81
posted on 2005-10-14 17:30 力为 阅读(7651) 评论(11)  编辑 收藏 引用 所属分类: 4. C++ FAQ

评论

# re: 如何通过FILE*操作内存文件? 2005-10-15 14:01 任我行
关注,等你写全了,我就来用,哈哈。
  回复  更多评论
  

# re: 如何通过FILE*操作内存文件? 2005-10-15 21:11 力为
这个在CSDN上放了几天也没有人回答
是不是太难了呢?  回复  更多评论
  

# re: 如何通过FILE*操作内存文件? 2005-10-17 15:50 moonriver
不理解为什么要这样干? 能否把BACKGROUND将清楚些!  回复  更多评论
  

# re: 如何通过FILE*操作内存文件? 2005-10-19 15:02 力为
background:
1. 使用第三方的接口生成某种格式的文件
2. 自己从该文件中分析需要的数据
3. 该文件为中间文件并不需要放在磁盘之上


故用内存文件保存临时数据
  回复  更多评论
  

# re: 如何通过FILE*操作内存文件? 2005-12-04 14:43 hehehe
不理解啊!把文件读入缓冲区,再处理缓冲区中的文件,是这个意思吗?  回复  更多评论
  

# re: 如何通过FILE*操作内存文件? 2005-12-05 09:14 力为
也可以这样认为。
只不过通过FILE如何把内容写入缓冲区?  回复  更多评论
  

# re: 如何通过FILE*操作内存文件? 2005-12-05 13:37 hehehe
一般来说,我比较喜欢这个函数。。。
BOOL ReadFile(
HANDLE hFile, // handle of file to read
LPVOID lpBuffer, // pointer to buffer that receives data
DWORD nNumberOfBytesToRead, // number of bytes to read
LPDWORD lpNumberOfBytesRead, // pointer to number of bytes read
LPOVERLAPPED lpOverlapped // pointer to structure for data
);
那个lpBuffer就是指向缓冲区的指针,要在内存中操作文件,我就用这个。。。  回复  更多评论
  

# re: 如何通过FILE*操作内存文件? 2005-12-05 13:50 力为
但C语言的FILE*如何往缓冲区写内容呢?  回复  更多评论
  

# re: 如何通过FILE*操作内存文件? 2006-01-07 10:27 力为
在悬挂了N月之后,我为此题下了结论:此题无解!:(  回复  更多评论
  

# re: 如何通过FILE*操作内存文件? 2006-01-22 15:20 stronger
https://sourceforge.net/project/shownotes.php?release_id=340318

  回复  更多评论
  

# re: 如何通过FILE*操作内存文件? 2010-10-08 01:44 macostar_wzy
内存映射API函数CreateFileMapping创建一个有名的共享内存:
HANDLE CreateFileMapping(
HANDLE hFile, // 映射文件的句柄,
//设为0xFFFFFFFF以创建一个进程间共享的对象
LPSECURITY_ATTRIBUTES lpFileMappingAttributes, // 安全属性
DWORD flProtect, // 保护方式
DWORD dwMaximumSizeHigh, //对象的大小
DWORD dwMaximumSizeLow,
LPCTSTR lpName // 必须为映射文件命名
);
要把文件映像到内存,首先必须调用CreateFileMapping()函数,它需要用一个由CreateFile()函数打开并返回的文件句柄,对大多数共享内存的应用程序,必须把此句柄设置为0xFFFFFFFF,用来指定系统页面文件。通过使用上面的特殊句柄,0XFFFFFFFF是INVALID_HANDLE_VALUE,表示不指定文件,在系统分页文件(pagefile.sys)中分配映射的磁盘空间,可以不调用CreateFile函数,当然在完成时,也不必有一个内存的磁盘文件拷贝。
  回复  更多评论
  


只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理