使用文件映射,我的映射文件中的数据在非法重启后,没了

myshinji 2009-08-24 03:52:54
CreateFileMapping
MapViewOfFile

使用了这两个函数,主要是为了像操作内存一样操作文件。

可是我在试验极端情况的时候,也即 非正常重启/意外断电 的时候,发现,我记录在映射文件中的内容都没了!一片空白。
我的程序是服务,我强行关闭进程的时候,文件里的内容是保存着的

谁知道怎么回事啊?什么方法可以保留这些内容?
...全文
140 18 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
18 条回复
切换为时间正序
请发表友善的回复…
发表回复
myshinji 2009-08-26
  • 打赏
  • 举报
回复
谢谢楼上各位关注这,问题自己解决了
我当时用的是FlushXXX,忘了,是flush文件句柄的,

但我应该用FlushViewOfFile(说实话是在别人的长篇文章中的一句话找到的函数),现在好了,试了10几次断电,都正常了。

------------------------------------
LS没有明白我的意思,没有来得及关闭文件,所以数据不会回写保存,可是我在写入“ABC”后,保存了,我又把“abc”改成“变成了BCD”,那么如果没关闭前就Reset了 我觉得你会同意:下次打开文件,内容仍是“ABC”。是吧?如果不同意的话,请讲解。

我的情况就是:连“ABC”都没了!!
myshinji 2009-08-25
  • 打赏
  • 举报
回复
晕了,难道又要无满意结贴。。
soliddream66 2009-08-25
  • 打赏
  • 举报
回复
[Quote=引用 12 楼 fly4free 的回复:]
晕了,我是在做软件,我的映射文件就是为了 防止用户数据在意外断电后恢复用的。

那还不如直接让客户买一个UPS呢……
[/Quote]
楼主,你的想法是有问题的,如果你一直保持打开文件状态,在断电瞬间,你怎么可能来得及关闭文件呢?

如果无法关闭文件,就这么强制断开的话,很显然会造成数据丢失,这是毫无疑问的。

要么,你就学word,在打开的同时,再生成一个缓存文件,这样的话,万一数据丢失,还可以从缓存文件恢复,当然你可以设置缓存文件为隐藏状态。

另外,能解决问题的方案就是好方案,没必要非得全用软件方式去解决,比如在这个问题中,我觉得用一个后备电源,一旦检测到供电方式不是来自交流电之后,立马关闭文件,并关闭电脑,这样的话文件数据就不会丢失了,你在BIOS设置一下power up,这样的话,一旦交流供电的话,电脑就自动开机。
soliddream66 2009-08-25
  • 打赏
  • 举报
回复
[Quote=引用 15 楼 soliddream66 的回复:]
引用 14 楼 myshinji 的回复:
晕了,难道又要无满意结贴。。

内存映射文件
  第一种办法是创建内存映射文件对象,然后将该对象映射到进程的地址空间中,再读取文件内容,然后倒序,再写入文件。

  第二种方法是,将文件内容读入一个大的缓冲区,然后倒序,再写入文件,中间对原来的文件删除,然后重新写入。

采用第二种方法
[/Quote]
采用I/O读写的缓存办法,而不是内存映射文件
soliddream66 2009-08-25
  • 打赏
  • 举报
回复
[Quote=引用 14 楼 myshinji 的回复:]
晕了,难道又要无满意结贴。。
[/Quote]
内存映射文件
  第一种办法是创建内存映射文件对象,然后将该对象映射到进程的地址空间中,再读取文件内容,然后倒序,再写入文件。

  第二种方法是,将文件内容读入一个大的缓冲区,然后倒序,再写入文件,中间对原来的文件删除,然后重新写入。

采用第二种方法
fly4free 2009-08-24
  • 打赏
  • 举报
回复
[Quote=引用 12 楼 fly4free 的回复:]
晕了,我是在做软件,我的映射文件就是为了 防止用户数据在意外断电后恢复用的。
[/Quote]

结果我的映射文件的内容也跟着没了,我白做了。。。
fly4free 2009-08-24
  • 打赏
  • 举报
回复
晕了,我是在做软件,我的映射文件就是为了 防止用户数据在意外断电后恢复用的。

那还不如直接让客户买一个UPS呢……
  • 打赏
  • 举报
回复
还没来得及写入磁盘文件中
soliddream66 2009-08-24
  • 打赏
  • 举报
回复
[Quote=引用 9 楼 fly4free 的回复:]
重要的是不是写不写如磁盘,问题是:“内容怎么全没了???”
[/Quote]
购买一个UPS后备电源可以解决这个问题!UPS可以在断电瞬间,自动切换后备电源给电脑供电!
fly4free 2009-08-24
  • 打赏
  • 举报
回复
重要的是不是写不写如磁盘,问题是:“内容怎么没了???”
whg01 2009-08-24
  • 打赏
  • 举报
回复
具体的更新机制不清楚,不过即使使用普通的文件操作方法也会存在类似的问题。执行了fwrite之后,不会立即写入磁盘,因为Windows对文件有缓存。只有fclose之后,文件才会百分百写入磁盘。
myshinji 2009-08-24
  • 打赏
  • 举报
回复
4楼,我就是存在文件里的啊?
命名文件,mytest.txt
myshinji 2009-08-24
  • 打赏
  • 举报
回复
我已确定:
在不是非法重启的时候,“映射文件一打开时”,是不会覆盖掉里面内容的,原来是什么仍然是什么

另外,我刚才试了几次,偶尔有几次试可以保留内容的。
哎,如果机制限定了是这样的,我就只能用纯文件操作了。。。

有没有人能够确定,到底是不是:在间隔刷新时间到了后,我通过MapViewOfFile得到的指针操作文件,这个时候机制导致了内容全部重新置0,然后就可以从内存中复制被映射的数据,但这个时候我按了重启,所以,数据丢失?而数据不丢失的时候,是因为,我按重启的时候没有进行指针的操作?

大侠指点一下~~~
whg01 2009-08-24
  • 打赏
  • 举报
回复
这个取决于Windows对内存映像文件的实现,强行关闭程序,Windows会释放程序使用的所有资源,所以内存映像文件在内存中的数据就被Windows写入到磁盘中。
直接重启则不会发生。
至于文件里全是0x00,我猜想可能是内存映像文件一打开时,会在磁盘上分配一组新的簇来保存数据,或者是在打开时就先把里面的数据写成0x00.你可以找个工具看一下。反正是Windows在背后偷偷干的。
MoXiaoRab 2009-08-24
  • 打赏
  • 举报
回复
呵呵。肯定没了。
虚存被释放掉了....过段时间就存在文件里一次保险下吧
myshinji 2009-08-24
  • 打赏
  • 举报
回复
对了,补充一下,我的确是每隔一段时间,刷新数据结构中的一个成员,也算和一楼说的一样了。

但是按机箱Reset后,重启,看到原来有数据的地方就都是0x00 了

之前有个数据一直是0x00,所以我感觉是没Flush的原因,后来,我unmapviewoffile,closehandle(文件映射句柄),然后再次重新建立映射的,我以为这样会类似flush功能,结果弄成了全没了。。。。

文件映射是永久存在的吗?我用的是实际的文件mytest.txt
myshinji 2009-08-24
  • 打赏
  • 举报
回复
请注意,我说的是“我记录在映射文件中的内容都没了!一片空白。

不是说数据不跟着刷新,而是,数据没了!WinHex查看,全是0x00,但应该是有数据的
贪玩的老鼠 2009-08-24
  • 打赏
  • 举报
回复
这个是没办法的,只能隔一个时间段,保存一下数据.
或者一有修改就立即保存(但这样就没有起到文件影射的优势).

16,548

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC相关问题讨论
社区管理员
  • 基础类社区
  • AIGC Browser
  • encoderlee
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

        VC/MFC社区版块或许是CSDN最“古老”的版块了,记忆之中,与CSDN的年龄几乎差不多。随着时间的推移,MFC技术渐渐的偏离了开发主流,若干年之后的今天,当我们面对着微软的这个经典之笔,内心充满着敬意,那些曾经的记忆,可以说代表着二十年前曾经的辉煌……
        向经典致敬,或许是老一代程序员内心里面难以释怀的感受。互联网大行其道的今天,我们期待着MFC技术能够恢复其曾经的辉煌,或许这个期待会永远成为一种“梦想”,或许一切皆有可能……
        我们希望这个版块可以很好的适配Web时代,期待更好的互联网技术能够使得MFC技术框架得以重现活力,……

试试用AI创作助手写篇文章吧