rename 修改文件名

Linux的 rename 命令有两个版本,一个是C语言版本的,一个是Perl语言版本的,早期的Linux发行版基本上使用的是C语言版本的,现在已经很难见到C语言版本的了,由于历史原因,在Perl语言大红大紫的时候,Linux的工具开发者们信仰Perl能取代C,所以大部分工具原来是C版本的都被Perl改写了,因为Perl版本的支持正则处理,所以功能更加强大,已经不再需要C语言版本的了。

如何区分系统里的rename命令是哪个版本的?

输入 man rename 看到第一行是

RENAME(1) Linux Programmer’s Manual RENAME(1)

那么 这个就是C语言版本的。

而如果出现的是:

RENAME(1)    Perl Programmers Reference Guide    RENAME(1)

这个就是Perl版本的了!

两个版本的语法差异:

C语言的,按照man上面的注解,

rename的语法格式是:

rename fromtofile

这个命令有三个参数,分别是 from : 修改什么名字, to:改成什么名字, file 需要修改的文件是哪些。

用法示例:

比如,有一批文件,都是以 log开头的, log001.txt,  log002.txt ……. 一直到 log100.txt

现在想要把这批文件的log全部替换为 history

rename  log history log*

这句命令的意思很明白了,把 以 log开头的所有文件中的 log字符替换为 history

这样替换后的文件是: history001.txt,  history002.txt ….. 一直到 history100.txt

rename C语言版本的另一个man示例是把后缀名批量修改,

比如我们要将所有 jpeg的后缀名图片文件修改为 jpg文件。

rename .jpeg.jpg*.jpeg

这样,所有以 .jpeg扩展的后缀名全部被修改为 .jpg

现在总结一下rename C语言版本所能实现的功能: 批量修改文件名,结果是每个文件会被用相同的一个字符串替换掉!也就是说,无法实现诸如循环 然后按编号重命名!

Perl 版本的批量重命名,带有Perl的好处是,你可以使用正则表达式来完成很奇特的功能。

perl 版本的参数格式:

rename  perlexprfiles

注意,perl版本的rename只有两个参数,第一个参数为perl正则表达式,第二个参数为所要处理的文件

man rename的帮助示例:

1) 有一批文件,以 .bak结尾,现在想把这些 .bak 统统去掉。

rename     ‘s/\.bak$//‘       *.bak
这个命令很简单,因为我还没有系统学习过perl,我不知道perl里替换字符串是不是这么干的,但sed是这么干的,所以如果你有sed或者tr基础,很容易明白,这个替换和sed里的正则语法是一模一样的。

2) 把所有文件名内含有大小字母的,修改为小写字母。

rename      ‘y/A-Z/a-z/‘      *
依然和sed的替换语法一样,不用多解释,如果看不懂的话,可以系统学习一下sed先。

还有几个比较实用的例子:

批量去掉文件名里的空格

Linux文件名本来是不支持空格的,不知道什么时候允许了,当然,在命令行调用文件的时候,空格是很有问题滴,比如你 原来可以直接  mv  oldfile  newfile  但有空格就不行了 , 得加双引号: mv “oldfile”  “newfile” 或者用反斜杠转移  \[] ,这样还好,但如果你直接把含有空格的图片名引入 Latex文档,Latex生成pdf的时候会直接打印出文件名,之前这个问题苦恼了我很久,我生成的pdf怎么老是出现文件名呢?后来才发现原来是文件名内含有空格的问题!windows系统下生成的文件名是天生含有空格的,虽然很讨厌,但有些惠普扫描仪生成的图片默认就加入了空格,没有办法,只好去掉他,在系统研究rename命令前,我是用 mv 去除空格的。

网上流程的两个去空格的版本:

1) tr 版:

复制代码

代码如下:

find . -type f -name "* *" -print |
while read name; do
na=$(echo $name | tr ‘ ‘ ‘_‘)
if [[ $name != $na ]]; then
mv "$name" $na
fi
done

这个版本以前我一直用的,不知道哪个网上搜刮来的,当时还没有系统的学习过 tr/sed/awk命令。

注解一下,很好理解, find . type f -name “* *” -print 这一句是查找当前目录下所有类型为普通文件的 并且名字之中含有空格的文件,并打印出来,其实 find默认就是打印的 这个 -print 多余了,然后 通过管道传输给 while 循环读取,文件名放到 name 变量里,用 tr 命令 替换空格为 下划线。 下面判断如果执行后的名称不相同,使用 mv 命令重命名。但这个if判断可有可无,因为find已经查询了所有文件名中含有空格的,那么经过 tr 命令后, $na变量肯定不等于 $name 变量的。

所以这段代码可以简化:

复制代码

代码如下:

find . -type f -name "* *" |
while read name; do
na=$(echo $name | tr ‘ ‘ ‘_‘)
mv "$name" "$na" 
done

tr 可以看着是 sed 的一个精简版本,tr 用下划线来替换空格。

还有一个 是 sed 版本实现:

for f in *;do mv "$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过程,并不直接运行,可以查看测试效果后,然后再运行。

时间: 2024-10-09 14:14:29

rename 修改文件名的相关文章

Linux下批量修改文件名(rename)

原文地址: http://blog.csdn.net/sea_shore/article/details/6102437 1.rename命令批量修改文件名, 其实linux下可以使用别的办法来批量修改文件名, 不过rename实在太方便了 比如把所有的表为cdb1_* 修改为cdb_*的在本目录下只需要# rename 'cdb1' 'cdb' * 以前都是写个for循环来做...想想多傻啊, 呵呵 rename还有更多的功能, 建议man rename下 From:http://www.ha

linux下批量修改文件名之rename

最近因为突然用到需匹配更换文件名,发现rename命令真是 简单好用,和sed语法及vim 替换很相似. 1. 更改文件名后缀 rename 's/\.txt/\.html/' * 2.增加文件名后缀 rename 's/$/\.txt/' * 3.删除文件名后缀 rename 's/.txt//' * 4.模式匹配修改文件名 rename 's/A-Z/a-z/' or rename 's/(\d)/第$1章/' *

Linux下批量修改文件名(rename)

1.rename命令批量修改文件名, 其实Linux下可以使用别的办法来批量修改文件名, 不过rename实在太方便了 比如把所有的表为cdb1_* 修改为cdb_*的在本目录下只需要# rename 'cdb1'  'cdb'  * 以前都是写个for循环来做...想想多傻啊, 呵呵 rename还有更多的功能, 建议man rename下 From:http://www.hao32.com/unix-linux/42.html 2.批量更改文件名 rename 通过 man rename 命

Linux中rename命令用法学习 修改文件名

如何用命令修改文件名呢?在linux下可以用rename命令,当然还可以使用mv命令,这里分享下linux rename命令的用法. 有一部分人说linux下没有rename命令,建议大家用mv命令.对rename命令和mv命令在重命名文件方面做一个比较,大家根据个人喜欢选择使用吧. mv命令,在man mv中对于mv命令的介绍:mv -move(rename) files 可以看到mv命令确实有重命名的功能,但是实际应用中,它只能对单个文件重命名,命令如下:mv [path/]oldfilen

一键修改文件名步骤及问题

在要修改的文件夹下面按shifit+右键,打开命令提示符 输入dir /b>rename.xls,列出文件目录 打开rename.xls,在B列修改文件名,删除A列中不需要更改的文件名 在C列输入="ren "&A1&" "&B1,向下拖动到所有单元格 复制C列到记事本,保存为ren.bat文件,双击执行 注意的问题: 被修改文件名和修改的文件名不能出现空格,空格应该以" "代替: 若确认命令无误,却仍然修改不了时,

如何用python从文中获取文件名再用正则表达式批量修改文件名

第零步:问题的提出 我在网上购买了星火英语的六级晨读美文100篇(六级早已高分飘过,不过很喜欢这些文章,买来重新品味),但是发现其文章的命名都为01.txt或10.txt等.为了便于检索需要修改文件名称. 第一步:从文件中取出文件名. 我发现txt文件的第一行为文件名,格式为如:Passage 3. Three Passions I Have Lived for, 后面还有几个换行符.写下如下代码进行第一步修改: import os x = 1 while x < 101: if x < 10

Linux中批量修改文件名

有时候我们有一个目录下面的文件名都需要重新命令 如: 以下有十个学生完成的脚本,现在需要将finish改为ing [[email protected] test]# touch student_{01..10}_finish.sh [[email protected] test]# ls student_* student_01_finish.sh  student_03_finish.sh  student_05_f student_02_finish.sh  student_04_finis

python修改文件名

最近下了<中国式英语口语纠错>里面的文件都是"tingvoa.com_cnusa043.mp3",MP3播放器不识别, 因此用python脚本写了一个rename的脚本 # -*- coding: utf-8 -*- import os def filerename(path): for file in os.listdir(path): #print file oldname = os.path.basename(file) #tingvoa.com_cnusa043.m

shell脚本:批量修改文件名(删除文件名中字符)

shell脚本:批量修改文件名(文件名中添加字符) 上一篇写过批量修改文件名(文件名中添加字符),工作中还存在这样的需求,批量修改文件名,删除文件名中的某些字符: 举例如下:批量改名,删除文件名中多余字符 目录下文件名为如下,要求去掉_finished. stu_102999_1_finished.jpg stu_102999_2_finished.jpg stu_102999_3_finished.jpg stu_102999_4_finished.jpg stu_102999_5_finis