ljbhander 2012-07-12
Linux的rename命令有两个版本,一个是C语言版本的,一个是Perl语言版本的,早期的Linux发行版基本上使用的是C语言版本的,现在已经很难见到C语言版本的了,由于历史原因,在Perl语言大红大紫的时候,Linux的工具开发者们信仰Perl能取代C,所以大部分工具原来是C版本的都被Perl改写了,因为Perl版本的支持正则处理,所以功能更加强大,已经不再需要C语言版本的了。
如何区分系统里的rename命令是哪个版本的?
输入manrename看到第一行是
RENAME(1)LinuxProgrammer’sManualRENAME(1)
那么这个就是C语言版本的。
而如果出现的是:
RENAME(1)PerlProgrammersReferenceGuideRENAME(1)
这个就是Perl版本的了!
两个版本的语法差异:
C语言的,按照man上面的注解,
rename的语法格式是:
renamefromtofile
这个命令有三个参数,分别是from:修改什么名字,to:改成什么名字,file需要修改的文件是哪些。
用法示例:
比如,有一批文件,都是以log开头的,log001.txt,log002.txt.......一直到log100.txt
现在想要把这批文件的log全部替换为history
renameloghistorylog*
这句命令的意思很明白了,把以log开头的所有文件中的log字符替换为history
这样替换后的文件是:history001.txt,history002.txt.....一直到history100.txt
renameC语言版本的另一个man示例是把后缀名批量修改,
比如我们要将所有jpeg的后缀名图片文件修改为jpg文件。
rename.jpeg.jpg*.jpeg
这样,所有以.jpeg扩展的后缀名全部被修改为.jpg
现在总结一下renameC语言版本所能实现的功能:批量修改文件名,结果是每个文件会被用相同的一个字符串替换掉!也就是说,无法实现诸如循环然后按编号重命名!
2,Perl版本的批量重命名,带有Perl的好处是,你可以使用正则表达式来完成很奇特的功能。
perl版本的参数格式:
renameperlexprfiles
注意,perl版本的rename只有两个参数,第一个参数为perl正则表达式,第二个参数为所要处理的文件
manrename的帮助示例:
1)有一批文件,以.bak结尾,现在想把这些.bak统统去掉。
rename's/\.bak$//'*.bak
这个命令很简单,因为我还没有系统学习过perl,我不知道perl里替换字符串是不是这么干的,但sed是这么干的,所以如果你有sed或者tr基础,很容易明白,这个替换和sed里的正则语法是一模一样的。
2)把所有文件名内含有大小字母的,修改为小写字母。
rename'y/A-Z/a-z/'*
依然和sed的替换语法一样,不用多解释,如果看不懂的话,可以系统学习一下sed先。
还有几个比较实用的例子:
1)批量去掉文件名里的空格
Linux文件名本来是不支持空格的,不知道什么时候允许了,当然,在命令行调用文件的时候,空格是很有问题滴,比如你原来可以直接mvoldfilenewfile但有空格就不行了,得加双引号:mv"oldfile""newfile"或者用反斜杠转移\[],这样还好,但如果你直接把含有空格的图片名引入Latex文档,Latex生成pdf的时候会直接打印出文件名,之前这个问题苦恼了我很久,我生成的pdf怎么老是出现文件名呢?后来才发现原来是文件名内含有空格的问题!windows系统下生成的文件名是天生含有空格的,虽然很讨厌,但有些惠普扫描仪生成的图片默认就加入了空格,没有办法,只好去掉他,在系统研究rename命令前,我是用mv去除空格的。
网上流程的两个去空格的版本:
1)tr版:
find.-typef-name"**"-print|
whilereadname;do
na=$(echo$name|tr'''_')
if[[$name!=$na]];then
mv"$name"$na
fi
done
这个版本以前我一直用的,不知道哪个网上搜刮来的,当时还没有系统的学习过tr/sed/awk命令。
注解一下,很好理解,find.typef-name"**"-print这一句是查找当前目录下所有类型为普通文件的并且名字之中含有空格的文件,并打印出来,其实find默认就是打印的这个-print多余了,然后通过管道传输给while循环读取,文件名放到name变量里,用tr命令替换空格为下划线。下面判断如果执行后的名称不相同,使用mv命令重命名。但这个if判断可有可无,因为find已经查询了所有文件名中含有空格的,那么经过tr命令后,$na变量肯定不等于$name变量的。
所以这段代码可以简化:
find.-typef-name"**"|
whilereadname;do
na=$(echo$name|tr'''_')
mv"$name""$na"
done
tr可以看着是sed的一个精简版本,tr用下划线来替换空格。
还有一个是sed版本实现:
forfin*;domv"$f"`echo"$f"|sed's/[]\+/_/g'`;done
这里的sed表达式还可以这样写:
sed's/[[:space:]]\+/_/g'
不过记住,sed里的出现一次或多次的加号是需要添加反斜杠的。即:\+
这样就可以了。
好了,这两种办法都太他妈罗嗦了,看看rename实现吧:
rename's/[]+/_/g'*
OK就这么简单。
方括号内的空格可以用[:space:]代替,
即可以写成's/[[:space:]]+/_/g'
这里注意,rename采用的是标准perl正则语法,所以无须将加号转变为反斜杠加号
即+不能修改为\+,否则替换失败。
还有几个好玩的例子:
比如统一在文件头部添加上hello
rename's/^/hello/'*
统一把.html扩展名修改为.htm
rename's/.html$/.htm/'*
统一在尾部追加.zip后缀:
rename's/$/.zip/'*
统一去掉.zip后缀:
rename's/.zip$//'*
规则化数字编号名,比如1.jpg,2.jpg.....100.jpg,现在要使文件名全部三位即1.jpg....001.jpg
运行两次命令:
rename's/^/00/'[0-9].jpg#这一步把1.jpg.....9.jpg变幻为001.jpg....009.jpg
rename's/^/0/'[0-9][0-9].jpg#这一步把10.jpg.....99.jpg变幻为010.jpg.....090.jpg
Ok,rename就研究了这么多,暂时不知道如何在rename中引入动态变量,比如$i++
我测试过i=0;rename-n"s/^.*$/$((++i))/"*执行后i被自增了1,并非想我想像中那样,可以在每操作一个文件自增一,猜想可能是因为rename批量实现的,导致++i只计算一次!
-n用来测试rename过程,并不直接运行,可以查看测试效果后,然后再运行。