LinuxのPythonでプロセスの開始時間(または稼働時間)を取得するにはどうすればよいですか?
私が知っているのは、「ps -p my_process_id -f」を呼び出して、出力を解析できることだけです。しかし、それはクールではありません。
psutil https://github.com/giampaolo/psutilを使用:
>>> import psutil, os, time
>>> p = psutil.Process(os.getpid())
>>> p.create_time()
1293678383.0799999
>>> time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(p.create_time()))
'2010-12-30 04:06:23'
>>>
...さらに、Linux だけでなく、クロスプラットフォームです。
注意: 私はこのプロジェクトの作成者の 1 人です。
測定しようとしている python プログラム内から実行している場合は、次のようにすることができます。
import time
# at the beginning of the script
startTime = time.time()
# ...
def getUptime():
"""
Returns the number of seconds since the program started.
"""
# do return startTime if you just want the process start time
return time.time() - startTime
ps
それ以外の場合は、解析するか、に入るしかありません/proc/pid
。bash
経過時間を取得する良い方法は次のとおりです。
ps -eo pid,etime | grep $YOUR_PID | awk '{print $2}'
これは経過時間を次の形式で出力するだけなので、解析は非常に簡単です。
days-HH:MM:SS
(実行時間が 1 日未満の場合は、単にHH:MM:SS
)
開始時間は次のように取得できます。
ps -eo pid,stime | grep $YOUR_PID | awk '{print $2}'
残念ながら、プロセスが今日開始されなかった場合、時間ではなく開始日のみが表示されます。
これを行う最善の方法は、経過時間と現在の時間を取得し、ちょっとした計算を行うことです。以下は、PID を引数として取り、上記の処理を実行して、プロセスの開始日時を出力する Python スクリプトです。
import sys
import datetime
import time
import subprocess
# call like this: python startTime.py $PID
pid = sys.argv[1]
proc = subprocess.Popen(['ps','-eo','pid,etime'], stdout=subprocess.PIPE)
# get data from stdout
proc.wait()
results = proc.stdout.readlines()
# parse data (should only be one)
for result in results:
try:
result.strip()
if result.split()[0] == pid:
pidInfo = result.split()[1]
# stop after the first one we find
break
except IndexError:
pass # ignore it
else:
# didn't find one
print "Process PID", pid, "doesn't seem to exist!"
sys.exit(0)
pidInfo = [result.split()[1] for result in results
if result.split()[0] == pid][0]
pidInfo = pidInfo.partition("-")
if pidInfo[1] == '-':
# there is a day
days = int(pidInfo[0])
rest = pidInfo[2].split(":")
hours = int(rest[0])
minutes = int(rest[1])
seconds = int(rest[2])
else:
days = 0
rest = pidInfo[0].split(":")
if len(rest) == 3:
hours = int(rest[0])
minutes = int(rest[1])
seconds = int(rest[2])
elif len(rest) == 2:
hours = 0
minutes = int(rest[0])
seconds = int(rest[1])
else:
hours = 0
minutes = 0
seconds = int(rest[0])
# get the start time
secondsSinceStart = days*24*3600 + hours*3600 + minutes*60 + seconds
# unix time (in seconds) of start
startTime = time.time() - secondsSinceStart
# final result
print "Process started on",
print datetime.datetime.fromtimestamp(startTime).strftime("%a %b %d at %I:%M:%S %p")
man proc
の22番目の項目/proc/my_process_id/stat
は次のとおりです。
starttime %lu
システムの起動後にプロセスが開始された時間。
ここでの問題は、jiffyの長さを決定する方法と、システムがいつ起動したかを決定する方法です。
後者の答えはまだから来ていman proc
ます:それは次/proc/stat
のようにそれ自身の行にあります:
btime 1270710844
これは、エポックからの秒単位の測定値です。
前者の答えはよくわかりません。man 7 time
言う:
ソフトウェアクロック、HZ、およびJiffies
多くのシステムコールとタイムスタンプの精度は、ソフトウェアクロックの解像度によって制限されます。ソフトウェアクロックは、カーネルによって維持され、瞬間的に時間を測定します。jiffyのサイズは、カーネル定数の値によって決まります
HZ
。の値は、HZ
カーネルのバージョンやハードウェアプラットフォームによって異なります。x86では、状況は次のようになります。2.4.xまでのカーネルでHZ
は、100であり、0.01秒のjiffy値が得られました。2.6.0から、HZ
1000に引き上げられ、0.001秒の瞬間が得られました。カーネル2.6.13以降、HZ
値はカーネル構成パラメーターであり、100、250(デフォルト)、または1000になり、それぞれ0.01、0.004、または0.001秒のjiffies値が生成されます。
を見つける必要がありますHZ
が、値が250であることを期待する以外は、Pythonからそれをどのように実行するかわかりません(ウィキペディアがデフォルトであると主張しているため)。
ps
このようにそれを取得します:
/* sysinfo.c init_libproc() */
if(linux_version_code > LINUX_VERSION(2, 4, 0)){
Hertz = find_elf_note(AT_CLKTCK);
//error handling
}
old_Hertz_hack(); //ugh
これは、Python用の非常に小さなCモジュールによってうまく行われた仕事のように聞こえます:)
badp の回答に基づくコードは次のとおりです。
import os
from time import time
HZ = os.sysconf(os.sysconf_names['SC_CLK_TCK'])
def proc_age_secs():
system_stats = open('/proc/stat').readlines()
process_stats = open('/proc/self/stat').read().split()
for line in system_stats:
if line.startswith('btime'):
boot_timestamp = int(line.split()[1])
age_from_boot_jiffies = int(process_stats[21])
age_from_boot_timestamp = age_from_boot_jiffies / HZ
age_timestamp = boot_timestamp + age_from_boot_timestamp
return time() - age_timestamp
それが正しいかどうかはわかりません。sleep(5) を呼び出して実行するテスト プログラムを作成しましたが、出力が間違っていて、実行ごとに数秒間変化します。これは vmware ワークステーション vm にあります。
if __name__ == '__main__':
from time import sleep
sleep(5)
print proc_age_secs()
出力は次のとおりです。
$ time python test.py
6.19169998169
real 0m5.063s
user 0m0.020s
sys 0m0.036s
def proc_starttime(pid=os.getpid()):
# https://gist.github.com/westhood/1073585
p = re.compile(r"^btime (\d+)$", re.MULTILINE)
with open("/proc/stat") as f:
m = p.search(f.read())
btime = int(m.groups()[0])
clk_tck = os.sysconf(os.sysconf_names["SC_CLK_TCK"])
with open("/proc/%d/stat" % pid) as f:
stime = int(f.read().split()[21]) / clk_tck
return datetime.fromtimestamp(btime + stime)
解析できます/proc/uptime
>>> uptime, idletime = [float(f) for f in open("/proc/uptime").read().split()]
>>> print uptime
29708.1
>>> print idletime
26484.45
Windowsマシンの場合、おそらくwmiを使用できます
import wmi
c = wmi.WMI()
secs_up = int([uptime.SystemUpTime for uptime in c.Win32_PerfFormattedData_PerfOS_System()][0])
hours_up = secs_up / 3600
print hours_up