在Unix / Linux中使用命令行跟踪时间的日期和时间计算

时间:2009-01-02 21:50:49

标签: bash

我正在寻找一种简单的方法来跟踪我使用命令行处理项目的时间。我做了

echo "$(date) : Start" >>worktime.txt

当我开始工作时

echo "$(date) : End" >>worktime.txt

当我停止工作时结果是以下形式的文件:

kent@rat:~/work$ cat worktime.txt 
Fri Jan  2 19:17:13 CET 2009 : Start
Fri Jan  2 19:47:18 CET 2009 : End
Fri Jan  2 21:41:07 CET 2009 : Start
Fri Jan  2 22:39:24 CET 2009 : End

“开始”和“结束”是指描述,可能是任何其他文字。 确定每个奇数行包含之前的开始日期和时间:并且每个偶数行包含结束日期和时间。

我想要做的是通过使用bash,tr,sed,awk和其他常见的Unix / Linux工具解析文本文件来总结我的工作时间。我更喜欢使用这种方法,而不是使用Python,Perl,Haskell,Java等。

欢迎所有想法!

5 个答案:

答案 0 :(得分:7)

假设您正在使用GNU coreutils,请改为运行date +%s

echo $(date +%s) : Start

%s占位符与非POSIX兼容,并且特定于GNU日期。它被自纪元以来经过的秒数所取代。接下来编写一个记录差异并加以总结的脚本。你会做类似

的事情
#!/bin/bash

total=0
while read LINE; do
  t1=$(echo $LINE |cut -d' ' -f1)
  read LINE
  t2=$(echo $LINE |cut -d' ' -f1)
  total=$(($total+$t2-$t1))
done
echo "$(echo "scale=2; $total/(60*60)" |bc -l)" hours

如果您调用脚本sum_time.sh,则可以使用

调用脚本

~$ ./sum_time.sh <worktime.txt

答案 1 :(得分:5)

wilhelmtell有一个很棒的脚本,如果您需要使用稍加修改的版本,以便继续使用$(date)而不是$(date +%s),这将以相同的方式工作,只需添加一行使用+%s

重新格式化时间
#!/bin/bash
total=0
while read LINE; do
  d1=$(echo $LINE |cut -c1-28);
  t1=$(date -d "$d1" +%s);
  read LINE
  d2=$(echo $LINE |cut -c1-28);
  t2=$(date -d "$d2" +%s);
  total=$(($total+$t2-$t1));
done
echo $(($total/(60*60))) hours

答案 2 :(得分:2)

以下awk脚本应该这样做(尽管没有经过测试):

BEGIN {
    total_secs = 0
    FS = OFS = ":"
}

function snip_it() {
    $NF = ""; $0 = substr($0, 1, length($0)-1);
}

!(NR % 2) {
    snip_it();
    ("date -d \"" $0 "\" +%s") | getline start_secs
}

NR % 2 {
    snip_it();
    ("date -d \"" $0 "\" +%s") | getline end_secs
    total_secs += (end_secs - start_secs)
}

END {
    print "Total work time of " total_secs " seconds"
}

关键是date +%s返回自1970年以来的秒数,-d指定替代时间/日期,而不是now

答案 3 :(得分:1)

或者你可以使用已存在的东西:

http://fossies.org/linux/misc/old/tt-1.0.tar.gz

答案 4 :(得分:1)

目标是进行如下整体计算:

total= (FirstEnd - FirstStart ) + (ScndEnd - ScndStart) 

可写:

total= - FirstStart + FirstEnd - ScndStart + ScndEnd ...

唯一的要求是worktime.txt中只有完成的对(每个开头都有相应的结尾)。

有一种非常快捷的方式:

echo $(sed <worktime.txt 's/ : .*$//g'|date -f - +%s|xargs printf "-%s+%s")|bc
5302

这个解决方案真的很快,因为根本只有3个fork,而不依赖于必须处理多少日期。所有的叉子只做一次。

这可以写成:

echo $(
    sed <worktime.txt 's/ : .*$//g' |
        date -f - +%s |
        xargs printf "-%s+%s"
  ) | bc