0

我有一个应用程序正在修改 5 个相同的 xml 文件,每个文件都位于不同的网络共享上。我知道这是不必要的多余,但“必须如此”。

每次运行此应用程序时,都会添加/删除/修改一个元素(不多也不少)。

最初,应用程序打开每个 xml 文件,将元素添加/删除/修改到适当的节点并保存文件,如果不能,则抛出错误(无法访问网络共享,超时等...)

我如何使这个原子?

我最初的假设是:

foreach (var path in NetworkPaths)
    if (!File.Exists(path)
        isAtomic = false;

if (isAtomic)
{
    //Do things
}

但我可以看到,仅此而已。有没有其他方法可以做到这一点,或者我可以指出一个方向?

4

3 回答 3

2

不幸的是,要真正做到“原子”是不可能的。我最好的建议是为此包装您自己的交易形式,这样您至少可以撤消更改。

我会做一些类似检查每个文件的事情 - 如果一个不存在,抛出。

备份每个文件,保存撤消所需的状态,或者如果它们不是很大,则将副本保存在内存中。做不到就扔。

进行编辑,然后保存文件。如果您在此处遇到故障,请尝试从每个备份中恢复。您需要在此处进行一些错误处理,以便在恢复所有备份之前不要抛出异常。恢复后,抛出异常。

至少通过这种方式,您更有可能不会只对单个文件进行更改。希望,如果您可以修改一个文件,您将能够从备份中恢复它/撤消您的修改。

于 2009-05-11T15:18:36.777 回答
1

我建议以下解决方案。

  • 尝试使用写锁打开所有文件。
    • 如果一个或多个失败,则中止。
    • 修改并刷新所有文件。
      • 如果一个或多个失败,则将已修改的回滚并再次刷新。
  • 关闭所有文件。

如果回滚失败......好吧......再试一次,再试一次,再试一次......并在不一致的状态下放弃。

如果您可以控制所有写入此文件的进程,则可以使用锁定文件实现简单的锁定机制。您甚至可以执行预写日志记录并在锁定文件中记录计划的更改。如果您的进程崩溃,下一个尝试修改文件的进程将检测到不完整的操作,并可以在进行一次修改之前继续它。

于 2009-05-11T15:25:59.417 回答
0

我将介绍文件的版本控制。您可以通过将后缀附加到文件名来轻松完成此操作。例如一个计数器变量。读者的过程如下:

  • 准备下一个版本的文件
  • 将其写入具有不同名称的临时文件。
  • 获取最高版本号
  • 将此版本加一
  • 将临时文件重命名为新文件
  • 删除旧文件(您可以保留其中的 2 个)

作为读者你做 - 找到最高版本的文件 - 阅读它

于 2013-09-04T19:10:00.890 回答