6

假设我要编写一个 Python 脚本来捕获KeyboardInterrupt异常,以便用户使用Ctrl+C安全地终止

但是,我不能将所有关键操作(如文件写入)放入catch块中,因为它依赖于局部变量并确保后续Ctrl+C无论如何都不会破坏它。

使用带有空 ( pass)try部分的 try-catch 块和该部分内的所有代码finally将此片段定义为可能不会在中途中断的“原子、中断安全代码”是否可行并且是一种好习惯?

例子:

try:
    with open("file.txt", "w") as f:
        for i in range(1000000):
            # imagine something useful that takes very long instead
            data = str(data ** (data ** data))
            try:
                pass
            finally:
                # ensure that this code is not interrupted to prevent file corruption:
                f.write(data)

except KeyboardInterrupt:
        print("User aborted, data created so far saved in file.txt")
        exit(0)

在此示例中,我不关心当前生成的数据字符串,即创建可能会中断并且不会触发写入。但是一旦开始写入,就必须完成,这就是我要确保的全部。此外,如果在 finally 子句中执行写入时发生异常(或 KeyboardInterrupt)会发生什么?

4

2 回答 2

5

在您的情况下,没有问题,因为文件写入是原子的,但是如果您有一些文件对象实现,那就更复杂了,您try-except来错地方了。您必须在写入周围进行异常处理:

try:
    f.write(data)
except:
    #do some action to restore file integrity
    raise

例如,如果您写入二进制数据,您可以执行以下操作:

filepos = f.tell()
try:
    f.write(data)
except:
    # remove the already written data
    f.seek(filepos)
    f.truncate()
    raise
于 2016-04-21T06:34:33.600 回答
5

代码输入仍然finally可以被中断。Python 对此不做任何保证。它只保证finallytry套件完成后或套件中出现异常时执行将切换到套件try。Atry只能处理在其范围内引发的异常,而不是在其范围之外,并且finally在该范围之外。

因此,在声明中使用是没有意义的。传球是无操作的,它永远不会被打断,但套件仍然很容易被打断。trypassfinally

你需要选择一种不同的技术。您可以写入一个单独的文件并在成功完成后将其移动到位;例如,操作系统保证文件移动是原子的。或者记录你最后一次成功的写入位置,如果下一次写入被中断,则将文件截断到该点。或者在你的文件中写一个成功记录的标记,这样读取就知道要忽略什么。

于 2016-04-21T06:24:37.133 回答