如果我使用subprocess.Popen打开文件,如何知道文件是否已关闭?

时间:2014-07-17 12:14:18

标签: python subprocess psutil

我在python中使用以下命令打开文件:

file = os.path.normpath(file_path)
phandler = subprocess.Popen(['open', '-W', file])

我正在投票

is_opened = phandler.poll() == None

查看进程是否已终止。如果使用cmd + Q终止进程,它会工作。但是,如果我使用cmd + w关闭文件,则无效。什么是确保文件不再打开的简单方法。

PS。从phandler,如果需要,我也可以使用phandler.pid获取进程Id。

如果没有其他方法可以做到这一点,我可能不得不使用像watchdog这样的库,它是iNotify的包装器,但如果我可以使用像psutils这样的库来实现这一点,那将会更好。

1 个答案:

答案 0 :(得分:0)

你正在寻找这样的东西:

from __future__ import print_function  # If in Python 2.x, import print function from Python 3.x

import os
import subprocess
import time

import psutil

def get_proc_status(pid):
    """Get the status of the process which has the specified process id."""

    proc_status = None
    try:
        proc_status = psutil.Process(pid).status()
    except psutil.NoSuchProcess as no_proc_exc:
        print(no_proc_exc)
    except psutil.ZombieProcess as zombie_proc_exc:  
        # For Python 3.0+ in Linux (and MacOS?).
        print(zombie_proc_exc)
    return proc_status

def open_file(app, file_path):
    """Open the specified file with the specified app, and wait for the file to be closed. """

    my_file = os.path.normpath(file_path)
    # phandler = subprocess.Popen(['open', '-W', my_file])
    # Must be for Macs. Doesn't work in Windows or Linux.
    phandler = subprocess.Popen([app, my_file])
    pid = phandler.pid

    pstatus = get_proc_status(pid)
    while pstatus is not None and pstatus != "zombie":
        # The allowed set of statuses might differ on your system.
        print("subprocess %s, current process status: %s." % (pid, pstatus))
        time.sleep(1)  # Wait 1 second.
        pstatus = get_proc_status(pid)
        # Get process status to check in 'while' clause at start of next loop iteration.

    if os.name == 'posix' and pstatus == "zombie":   # Handle zombie processes in Linux (and MacOS?).
        print("subprocess %s, near-final process status: %s." % (pid, pstatus))
        phandler.communicate()

    pstatus = get_proc_status(pid)
    print("subprocess %s, final process status: %s.\n" % (pid, pstatus))

open_file(r"Evince\bin\Evince.exe", "fuzzed_file.pdf")
open_file(r"Evince\bin\Evince.exe", "unfuzzed_file.pdf")

根据Python:Non-Blocking + Non defunct processhow to kill (or avoid) zombie processes with subprocess module,似乎需要通信()来处理" zombie" Linux中的进程(我认为,在MacOS中)。

(参见StackOverflow' s Tag Info for zombie-process tag。)

这会生成输出,例如:

subprocess 6844, current process status: running.
subprocess 6844, current process status: running.
subprocess 6844, current process status: running.
subprocess 6844, current process status: running.
psutil.NoSuchProcess no process found with pid 6844
psutil.NoSuchProcess no process found with pid 6844
subprocess 6844, final process status: None.

subprocess 5460, current process status: running.
subprocess 5460, current process status: running.
subprocess 5460, current process status: running.
subprocess 5460, current process status: running.
subprocess 5460, current process status: running.
subprocess 5460, current process status: running.
psutil.NoSuchProcess no process found with pid 5460
psutil.NoSuchProcess no process found with pid 5460
subprocess 5460, final process status: None.