玩linux编程,文件对比还是经常用的;
[root@node1 ~]# tree directory1 directory2 directory1 ├── 1.png ├── 2.png └── 3.png directory2 ├── 2.png ├── 3.png └── 4.png
diff -r directory1 directory2
但是diff会对每个文件中的每一行都做比较,所以文件较多或者文件较大的时候会非常慢。请谨慎使用。
diff <(tree -Ci --noreport /mnt/f/自然马) <(tree -Ci --noreport /mnt/i/自然马) < /mnt/f/自然马 --- > /mnt/i/自然马 87a88 > xyz.avi 488d488 < rsync.txt 534d533 < 542D0.mp4
说明:
-C
选项是输出颜色,如果只是看一下目录的不同,可以使用该选项,但在结合其他命令使用的时候建议不要使用该选项,因为颜色也会转换为对应的编码而输出;-i
是不缩进,建议不要省略-i
,否则diff的结果很难看,也不好继续后续的文件操作;--noreport
是不输出报告结果,建议不要省略该选项。find directory1 -printf "%Pn" | sort > file1 find directory2 -printf "%Pn" | sort | diff file1 - 2d1 < 1.png 4a4 > 4.png
说明:
<
代表的行是directory1中有而directory2没有的文件,>
则相反,是directory2中有而directory1中没有。-printf "%Pn"
,此处的%P表示find的结果中去掉前缀路径,详细内容man find
。例如,find /root/ -printf "%Pn"
的结果中将显示/root/a/xyz.txt中去掉/root/后的结果:a/xyz.txt。如果不想使用-printf
,那么先进入各目录再find也行。
[root@node1 ~]# (cd /root/a;find . | sort >/tmp/file1) [root@node1 ~]# (cd /root/b;find . | sort | diff /tmp/file1 -) 2d1 < ./1.png 4a4 > ./4.png
上面将命令放进括号中执行是为了在子shell中切换目录,不用影响当前所在目录。
rsync -rvn --delete directory1/ directory2 | sed -n '2,/^$/{/^$/!p}' deleting a/xyz.avi rsync.txt 新建文件夹/542D0.mp4
其中deleting所在的行就是directory2中多出的文件。其他的都是directory中多出的文件。
如果想区分出不同的是目录还是文件。可以加上"-i"选项。
rsync -rvn -i --delete directory1/ directory2 | sed -n '2,/^$/{/^$/!p}' *deleting a/xyz.avi >f+++++++++ rsync.txt >f+++++++++ 新建文件夹/542D0.mp4
其中>f+++++++++
中的f代表的是文件,d代表的目录。
上面的rsync比较目录有几点要说明:
fileA-->fileB
的输出。vimdiff <(cd directory1; find . | sort) <(cd directory2; find . | sort) # 或者 vimdiff <(find directory1 -printf "%Pn"| sort) <(find directory2 -printf "%Pn"| sort)
meld是python写的一个图形化文件/目录比较工具,所以必须先安装图形界面或设置好图形界面接受协议。它的功能非常丰富,和win下的beyond compare有异曲同工之妙。
meld具体的使用方式就不介绍了。
个人建议使用命令行输出的结果中的方法方法三和方法四,因为它们都能很好地保留目录前缀。
以方法三为例:
find directory1 -printf "%Pn" | sort > file1 find directory2 -printf "%Pn" | sort | diff file1 -
以下是实验所需目录结构:
[root@node1 ~]# tree /root/a;tree /root/b /root/a ├── 1.png ├── 2.png └── 3.png 0 directories, 3 files /root/b ├── 2.png ├── 3.png ├── 4.png └── xen └── scripts └── block-drbd
首先比较这两个目录得到文件列表的差异。
find /root/a -printf "%Pn" | sort > /tmp/file1 find /root/b -printf "%Pn" | sort | diff /tmp/file1 - >diff.txt
然后从diff.txt中过滤出/root/a中多出的文件和/root/b中多出的文件。
# /root/a中多出的文件 awk '/</{printf("%s%sn","/tmp/etc/",$2)}' diff.txt /tmp/etc/1.png # /root/b中多出的文件 awk '/>/{printf("%s%sn","/tmp/etc/",$2)}' diff.txt /tmp/etc/4.png /tmp/etc/xen /tmp/etc/xen/scripts /tmp/etc/xen/scripts/block-drbd
需要注意的是,如果多了某个目录,则这个目录和其内所有文件都会列出来。如果要将多出的文件复制到其他地方,应当要注意这一点。
如果只想要比较出/root/a和/root/b下的文件和目录的不同,不再递归到子目录中比较。那么可以在find上继续加工一番:
find /root/a -maxdepth 1 -printf "%Pn" | sort > /tmp/file1 find /root/b -maxdepth 1 -printf "%Pn" | sort | diff /tmp/file1 - >diff.txt # /root/a中多出的文件 awk '/</{printf("%s%sn","/tmp/etc/",$2)}' diff.txt /tmp/etc/1.png # /root/b中多出的文件 awk '/>/{printf("%s%sn","/tmp/etc/",$2)}' diff.txt /tmp/etc/4.png /tmp/etc/xen
这样一来,/root/b中多出的文件就是4.png和xen,xen目录中的文件不再列出。