功能:tail命令可以输出文件的尾部内容,默认情况下它显示文件的最后十行。
显示每个指定文件的最后10 行到标准输出。若指定了多于一个文件,程序会在每段输出的开始添加相应文件名作为头。如果不指定文件或文件为"-" ,则从标准输入读取数据。
它常用来动态监视文件的尾部内容的增长情况,比如用来监视日志文件的变化。
语法:tail [选项] [文件]
短选项 | 长选项 | 涵义 |
---|---|---|
-c[+] K | --bytes=[+] K | 输出最后 K 字节;另外,使用-c +K 从每个文件的第 K 字节输出 |
-n[+] K | --lines=[+] K | 输出最后 K 行,代替最后10 行;使用-n +K 从每个文件的第 K 字节输出 |
-f | --follow=descriptor --follow=name |
descriptor是 --follow 默认值,所以 -f 等价 --follow 等价--follow=descriptor 即时输出文件变化后追加的数据。 tail -f file 动态跟踪文件file的增长情况,tail会每隔一秒去检查一下文件是否增加新的内容。如果增加就追加在原来的输出后面显示。但这种情况,必须保证在执行tail命令时,文件已经存在。如果想终止 tail -f 输出,按 Ctrl+C 中断tail程序。如果按Ctrl+C不能中断输出,那么可以在别的终端上执行 killall tail 强行终止。 |
--pid=PID | 同 -f 使用,当 PID 所对应的进程死去后,终止。 | |
---retry | 即使目标文件不可访问依然试图打开;在与参数-f 或 --follow=name 同时使用时常常有用。 | |
-F | --follow=name --retry | 与 -f 相同,也是动态跟踪文件的变化,不同的是执行此命令时文件可以不存在。 |
--max-unchanged-stats=N |
N 默认为5,使用--follow=name,重新打开一个在 N 次迭代后没有改变大小的文件来看它是否被解除连接或重命名(这是循环日志文件的通常情况)。 由于有inotify,这个选项很少使用。 |
|
-s S | --sleep-interval=S | 与 -f 合用,表示在每次反复的间隔休眠 S 秒 对于 --pid=PID ,每隔 S 秒检查进程。 |
-q | --quiet 或 --silent | 不输出文件名 |
-v | --verbose | 总是输出给出文件名的首部 |
-
K 后面可以跟乘号:
b 512, kB 1000, K 1024, MB 1000*1000, M 1024*1024,GB 1000*1000*1000, G 1024*1024*1024, 对于T, P, E, Z, Y 同样适用。 - 如果 + K (字节或者行数),那么从每个文件的开头算起的第 K 项开始显示。否则,显示该文件的最后 K 项。
- 如果您希望即时追查一个文件的有效名称而非描述内容(例如循环日志),默认的程序动作(--follow=descriptor)并不如您所愿。在这种场合可以使用 --follow=name 选项,它会使tail定期追踪打开给定名称的文件,以确认它是否被删除或被其它某些程序重新创建过。
- 同时按键盘上 Ctrl 键 和 C 键,退出显示。
tail 实例
测试的文件内容如下
[quietheart@lv-k tail_test]$cat tail_test
1 04_libraryTest
2 a.out
3 a.out.symbol
4 autotools
5 core.31058
6 cpp_test
7 download_blog
8 gpg_test
9 hello-1.0.tar.gz
10 hello-2.0
11 hello-2.0.tgz
12 main.cpp
13 mysql20110512.sql
14 rsynctest
15 tail_
16 tail_test
显示文件的最后10行
[quietheart@lv-k tail_test]$tail tail_test
显示文件tail_test的最后10行
7 download_blog
8 gpg_test
9 hello-1.0.tar.gz
10 hello-2.0
11 hello-2.0.tgz
12 main.cpp
13 mysql20110512.sql
14 rsynctest
15 tail_
16 tail_test
显示文件的最后N行
[quietheart@lv-k tail_test]$tail -n 5 tail_test
显示文件tail_test的最后5行
12 main.cpp
13 mysql20110512.sql
14 rsynctest
15 tail_
16 tail_test
从第5行开始显示文件
[quietheart@lv-k tail_test]$tail -n +5 tail_test
注意指定的数字前面 + 代表从这个数字相应的位置开始,显示后面的内容
5 core.31058
6 cpp_test
7 download_blog
8 gpg_test
9 hello-1.0.tar.gz
10 hello-2.0
11 hello-2.0.tgz
12 main.cpp
13 mysql20110512.sql
14 rsynctest
15 tail_
16 tail_test
显示文件最后38个字节的内容
[quietheart@lv-k tail_test]$tail -c 38 tail_test
ynctest
15 tail_
16 tail_test
显示文件第38个字节开始之后的内容
[quietheart@lv-k tail_test]$tail -c +38 tail_test
3 a.out.symbol
4 autotools
5 core.31058
6 cpp_test
7 download_blog
8 gpg_test
9 hello-1.0.tar.gz
10 hello-2.0
11 hello-2.0.tgz
12 main.cpp
13 mysql20110512.sql
14 rsynctest
15 tail_
16 tail_test注意指定的数字前面 + 代表从这个数字相应的位置开始,显示后面的内容。
跟踪文件,tail -f 两种模式
默认是以文件描述符方式,跟踪文件的增长
[quietheart@lv-k tail_test]$tail -f tail_test
此时等价于命令--follow=descriptor tail_test
7 download_blog
8 gpg_test
9 hello-1.0.tar.gz
10 hello-2.0
11 hello-2.0.tgz
12 main.cpp
13 mysql20110512.sql
14 rsynctest
15 tail_
16 tail_test
这里,输入之后,显示默认的文件的后10行,但是tail并没有因此而结束,而是一直在运行着,保持这那个文件对应的索引节点的打开状态。 接下来
如果使用echo new >>tail_test
,向文件追加新内容。那么会看到tail又继续将追加的内容打印出来。
如果使用echo new > tail_test
,清空原内容并重新写入新内容。那么tail会输出类似"tail: tail_test: file truncated"的字样来告诉文件内容被truncated了。
这个命令用于:跟踪动态增长的文件。例如系统日志。在默认情况下,根据它自己的文件描述符号来跟踪文件。
但是,有的程序追加文件内容的时候会将文件删除然后新建立一个。例如有些日志程序会在一定的时候将追加的日志文件重命名,然后再建立一个之前同名的新日志文件追加新的内容,这样的话这个命令就不好用了。再例如有些编辑器例如vim进行修改文件的时候,无法跟踪其变化。通过"ls -il"对vim编辑之前的文件和之后的文件的inode号对比发现,两者不同,应当是编辑的时候先删除文件的索引节点再新建立一个,新建的节点内容才包含了最新的内容,而之前tail打开的那个索引节点已经被删除了,看不见了,所以当然不会发生变化。
如果想要确定那么就先用"ps -aux |grep tail"找到tail的进程号,进入/proc目录的tail进程号目录中,查看其fd文件中的某个描述符号,例如"cat 3"这样会发现原来的内容。
以文件名方式,跟踪文件增长
[quietheart@lv-k tail_test]$tail --follow=name tail_test
7 download_blog
8 gpg_test
9 hello-1.0.tar.gz
10 hello-2.0
11 hello-2.0.tgz
12 main.cpp
13 mysql20110512.sql
14 rsynctest
15 tail_
16 tail_test
这样,tail根据文件名称跟踪文件的变化,默认来说tail就根据它自己的文件描述符号来跟踪文件,就像前面所说的,有的程序追加文件内容的时候会将文件删除然后新建立一个,例如有些日志程序会在一定的时候将追加的日志文件重命名,然后再建立一个之前同名的新日志文件追加新的内容,那么默认的方式就无法跟踪到文件的变化了,因为文件描述符号是代表一个索引节点的,而新追加的内容可能追加到新的索引节点上面了,这个时候就使用这个 --follow=name 选项。这样,如果当文件新追加内容是追加到同一名称的不同索引节点的情况发生时,那么tail那里就会提示 "tail: “tail_test” has been replaced; following end of new file"之后,再重新显示追加之后的新的最后10行。
tail -f 与 tail -F
tail -f 和tail -F 的区别
-f 参数,如果在追踪此文档时,此文档被删除、转移或者重建了, 那就停止不会再输出了。
-F 参数,如果在追踪此文档时,此文档被删除、转移或者重建了, 那会再重新try那個同名的那個文档, 如果重建了, 会继续追踪此文档。
tail -f 文件
-f 是--follow[=HOW]的缩写。"[=HOW]"有两个写法,一个"=descriptor",另一个是"=name"。[=HOW]省略时,默认使用的是"--follow=descriptor"。-f 等价 -follow 等价--follow=descriptor。第一个窗口
[root@cftest2 ~]#tail -f messages.3
helll test2
第二个窗口
[root@cftest2 ~]#rm messages.3
rm: remove regular file `messages.3'? y
[root@cftest2 ~]# echo "helll test3">>messages.3但是第一个窗口的tail -f 命令不会出现 hello test3
descriptor 虽然是默认的参数,但是不一定是最有用的。比如在tail 一个log文件的时候,这个文件很可能是按照日期或者大小滚动,文件滚动之后这个tail -f 命令,就失效了。
如果你跟踪的文件被被删除、转移或者重建, 你还想继续tail它, 你可以使用这个 tail --follow=name 或者 tail -F
tail -F 文件
-F 是 -follow=name --retry 的缩写。--follow=name 是按照文件名跟踪文件,可以定期去重新打开文件检查文件是否被其它程序删除并重新建立。 --retry 这个参数,保证文件重新建立后,可以继续被跟踪。
第一个窗口
[root@cftest2 ~]#tail -F messages.3
helll test1
tail: `messages.3' has become inaccessible: No such file or directory
tail: `messages.3' has appeared; following end of new file
helll test2
第二个窗口
[root@cftest2 ~]#rm messages.3
rm: remove regular file `messages.3'? y
[root@cftest2 ~]# echo "helll test3">>messages.3
第一个窗口可以看到,中间删除了messages.3,但重新创建后并输入helll test2,会继续显示出来。
tail -F 更强大
tail -F功能的强大,它等同于--follow=name --retry。如果你跟踪的文件被移动或者改名后, 你还想继续tail它, 你可以使用这个选项。
tail手册页中关于--retry的说明:keep trying to open a file even if it is inaccessible when tail starts or if it becomes inaccessible later; useful when following by name, i.e., with --follow=name。 tail命令开始执行时文件不存在或者执行过程中文件不能访问,会不断重试。
关于--follow的说明:-f, --follow[={name|descriptor}] output appended data as the file grows; -f, --follow, and --follow=descriptor are equivalent 。--follow=descriptor表明跟踪的是文件描述符,--follow=name表明跟踪的是文件名称。
如果文件名称改掉之后,还想继续跟踪原文件名称对应的尾部内容,就得使用 -F 选项而不是 -f 选项了。
[root@web imx_server]#tail -F log/IMX.LOG
14:13:28.892 INFO ImxConnection[6] imx.server.ImxConnection - RX IMX_ACTIVE_TEST{seq=3460,client_id=1291343201649042,presence_status=1(presence_status_online),}
14:13:28.892 DEBUG ImxConnection[6] org.logicalcobwebs.proxool.ImxDB - 006417 (01/02/00) - Connection #9 served
14:13:28.892 INFO ImxConnection[6] imx.dbo.ImxOnlineInfoRow - EXEC SQL UPDATE imx_online_info SET last_active_time = '2010-12-03 14:13:28.0' WHERE account = 'zhy'
14:13:28.894 DEBUG ImxConnection[6] org.logicalcobwebs.proxool.ImxDB - UPDATE imx_online_info SET last_active_time = '2010-12-03 14:13:28.0' WHERE account = 'zhy'; (1 milliseconds)
14:13:28.894 DEBUG ImxConnection[6] org.logicalcobwebs.proxool.ImxDB - 006417 (00/02/00) - Connection #9 returned (now AVAILABLE)
14:13:29.625 INFO ImxConnection[6] imx.server.ImxConnection - RX IMX_ACTIVE_TEST{seq=3461,client_id=1291343201649042,presence_status=1(presence_status_online),}
14:13:29.626 DEBUG ImxConnection[6] org.logicalcobwebs.proxool.ImxDB - 006418 (01/02/00) - Connection #8 served
14:13:29.626 INFO ImxConnection[6] imx.dbo.ImxOnlineInfoRow - EXEC SQL UPDATE imx_online_info SET last_active_time = '2010-12-03 14:13:29.0' WHERE account = 'zhy'
14:13:29.627 DEBUG ImxConnection[6] org.logicalcobwebs.proxool.ImxDB - UPDATE imx_online_info SET last_active_time = '2010-12-03 14:13:29.0' WHERE account = 'zhy'; (0 milliseconds)
14:13:29.653 DEBUG ImxConnection[6] org.logicalcobwebs.proxool.ImxDB - 006418 (00/02/00) - Connection #8 returned (now AVAILABLE)Ctrl+C
[root@web imx_server]#
总结一下:要想跟踪会更名的日志的话,用tail -F 而不是tail -f