Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1172334
  • 博文数量: 115
  • 博客积分: 950
  • 博客等级: 准尉
  • 技术积分: 1734
  • 用 户 组: 普通用户
  • 注册时间: 2011-12-08 20:46
文章分类

全部博文(115)

文章存档

2015年(5)

2014年(28)

2013年(42)

2012年(40)

分类: LINUX

2014-08-07 10:06:57

修改文件名可以有不同的命令方式,比如rename, mv都可以实现

对于单个的文件,可以直接使用以上的命令,那如果有大量的类似格式的文件名需要修改成其他格式的,该如何呢?

比如某次测试后,保存的文件为 Lan1.txt, Lan2.txt,....Lan100.txt

这一百个文件需要在前面添加前缀变成类似 ch7_Lan1.txt,如果你还想使用mv来一个一个...多痛苦啊

当前文件夹下,文件修改有4种方式

1. 使用while Loop加 ${//}来实现Lan到 ch7_Lan的替换

点击(此处)折叠或打开

  1. find . -name "Lan*txt" -type f | read files
  2. do
  3.   new=${files/Lan/ch7_Lan}
  4.   mv $files $new
  5. done

2. 充分利用 awk的分隔符功能来实现    

点击(此处)折叠或打开

  1. ls *Lan* | awk 'BEGIN{FS="Lan";OFS="ch7_Lan"}{printf "mv "$0" ";$1=$1;print $0}' | sh
首先,修改 FS和 OFS,输入和输出的分隔符
其次,使用 awk中的命令,拼接 “mv Lan1.txt ch7_Lan1.txt”这条命令
最后,使用|sh来执行一个 shell命令来完成 mv动作

其中,$1=$1是必须的,这个是完成 Lan===> ch7_Lan的关键(其实是可以使用 $2=$2等来替换,只要一个赋值动作而已)
奇怪的是,难道没有赋值动作,分隔符的改变就不能检测出来么??

3. 使用 awk的内置命令,gsub和 system来实现替换和命令执行

点击(此处)折叠或打开

  1. ls "*Lan*" | awk '{org=$0;gsub("Lan", "ch7_Lan");system("mv "org" "$1)}'
首先保存原始的数据
其次修改 Lan为 ch_Lan
最后执行mv动作

其中,最后system命令”mv “org” “需要使用”“来标记,
而且 org也需要”“,并且不能使用$org
1. 对于 awk命令中的变量,不需要用 $来引用
2. 在system()中,变量需要使用""来标识,否则就被当成字符串来处理
3. 如果对变量进行 $来进行引用,就会出现两种情况
   3.1 正常情况下,可以被当成位置参数来引用。比如 NF表示当前行的记录个数。比如echo "a b c d" | awk '{print NF, $NF}',这个会打印两个变量,第一个是 NF本身,其值为4;第二个是第四个位置参数,也就是 $4,也就是 d
   3.2 非正常情况下:
      3.2.1 如果变量本身就是个字符串。比如 echo "a b c d" | awk '{va=“varA";print va, $va}',因为后一个参数$va,由于va是个字符串,因此 $va打印的就是 $0
       3.2.2 如果变量本身就是个数字,但是该值大于 NF。 比如echo "a b c d" | awk '{va="varA";vb=6;print va, $va, vb, $vb, 1}',因为 $vb相当于是 $6,但是位置参数 $6没有,因此打印的就是个空字符

点击(此处)折叠或打开

  1. [martin@TSPerPacketsTest]$ echo "a b c d" | awk '{print NF, $NF}'
  2. 4 d
  3. [martin@TSPerPacketsTest]$ echo "a b c d" | awk '{va="varA";print va, $va}'
  4. varA a b c d
  5. [martin@TSPerPacketsTest]$ echo "a b c d" | awk '{va="varA";vb=6;print va, $va, vb, $vb, 1}'
  6. varA a b c d 6  1

4.  使用 sed中的 s来进行替换,然后使用e命令来执行

点击(此处)折叠或打开

  1. [martin@TSPerPacketsTest]$ ls *Lan* | sed -r -n 's/(.*)Lan(.*)/mv & \1ch_Lan\2/e'
关于sed的选项,使用 -r和 -n
-r 来启用后续可以使用 command,比如 mv
-n 来关闭 patten space中内容的显示

关于sed的 command,使用e。如此在使用s命令完成 pattern space中 Lan到 ch7_Lan的修改后,启动shell来执行 pattern space中的 mv指令
如果这里使用 p,标记仅仅显示  pattern space中的指令而已


如果对于某目录下的所有文件,进行文件名修改,有3中方式

1. 汇集awk的 FS/OFS和 gsub/system来实现。有点类似于综合上述2,3两种方式

点击(此处)折叠或打开

  1. [martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | awk 'BEGIN{FS="/";OFS="/"}{org=$0;gsub("Lan", "ch7_Lan", $NF);system("echo "org" "$0)}'
为了实现所有目录下的文件名进行修改,但是又不影响路径上文件夹名
首先,标记FS和OFS都为 ”/“
其次,使用gsub时候,指定仅仅修改当前行中最后一个记录,使用 $NF来指定

当然,对于FS和 OFS的指定,可以不放置在 BEGIN中实现,比如:

点击(此处)折叠或打开

  1. [martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | awk -vF=/ -vOFS=/ '{org=$0;gsub("Lan", "ch7_Lan", $NF);system("echo "org" "$0)}'
  2. [martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | awk -F/ -vOFS=/ '{org=$0;gsub("Lan", "ch7_Lan", $NF);system("echo "org" "$0)}'

2. 或者使用 sed

点击(此处)折叠或打开

  1. [martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | sed -r -n 's/(.*)([^\/]*)Lan([^\/]*)$/mv & \1\2ch7_Lan\3/e'

其中的各个 option和 command参数,上述都已经有介绍了。
这里,针对文件夹名和文件名,使用了简单的sed内置的正则表达式来进行匹配而已,来实现仅仅对文件名的修改


以上的实现方式,参考CU的一个讨论帖子:
阅读(26708) | 评论(0) | 转发(2) |
给主人留下些什么吧!~~