我找了段perl入门里的一段代码。。感觉这个更简单 9.5.3 更新大量文件 更新文件最常用的方法是写一个和以前的文件相似的新文件,我们可以根据的需要进行修改。如你所知,这和对同一个文 件上进行更新的结果类似,但上述方法有一些副作用。 本例中,我们有相似格式的上百个文件。其中一个是fred03.dat,如下: Promram name: granite Author: Gilbert Bates Company: RockSoft Department: R&D Phone: +1 503 555-0095 Date: Tues March 9, 2004 Version: 2.1 Size: 21k Status: Final beta 我们希望修改这个文件,使之含有不同的信息。下面是我们希望修改后它所呈现的样式: Program name: granite Author: Randal L. Schwartz Company: RockSoft Department: R&D Date: June 12, 2008 6:38 pm Version: 2.1 Size: 21k Status: Final beta 简言之,我们需要在3 个地方进行修改。作者(Author)的名字需要更改,日期(Date)需要更新,电话(Phone)需要删除。 我们需要在上百个这样的文件中作这些修改。 Perl 可以通过尖括号操作符(<>)对文件进行修改。下面程序能完成我们希望的工作,虽然第一次看时,不是很明显。这个 Perl 语言入门(第四版) [email protected] 118 / 201 9/21/2006 程序只有一个新的特性:特殊变量$^I;现在可以不用管它,我们将在后面讨论: #! /usr/bin/perl -w use strict; chomp(my $date = 'date'); $^I =".bak"; while(<>){ s/^Author:.*/Author: Randal L. Scharwartz/; s/^Phone:.*\n//; s/^Date:.*/Date: $date/; print; } 由于需要当前的日期,因此在程序开端使用了系统命令:date。另一种获得时间的更好的方法是(格式有些不同)使用Perl 自带的localtime 函数,其在标量context 中使用: my $date = localtime; 下一行是给$^I 赋值,我们现在不讨论它。 根据我们现在所知的,上述操作的结果是文件中新修改的部分被输出到终端,内容快速滚动,但文件本身不会被修改。 由尖括号操作(<>)所得到的文件列表来源于命令行。主循环读入,更新,输出每一行。(根据我们现在所知的,上述操作 的结果是文件中被修改的部分被输出到终端,这些内容快速滚动,但文件本身不会被修改。)第二个替换操作将含有电话 (phone)号码的整行由空串替换,连换行符一起替换掉。这行输出时,什么也不会出现,就像电话(Phone)号码从没存在过一 样。还有大量的行不会被这三个模式所匹配,他们在输出时不会有任何更改。 结果接近于我们所期望的了,除了还不知道怎样将更新的信息写回文件。答案是变量$^I。默认时为undef,此时没有什么 特殊的地方。但给它设置某些串时,它使尖括号操作(<>)变得有些特殊。 我们知道尖括号(<>)的神奇特点:如果没有指定文件名,则其从标准输入流中自动打开和关闭一系列文件进行读入。但如 果$^I 中有字符串,这个字符串则会成为备份文件的扩展名。我们在下面仔细讨论。 我们假设此时尖括号(<>)打开的文件是fred03.dat。它像以前那样打开它,但进行了重名名,把它叫做fred03.dat.bak◆。这 很好,因为不在使用之前的名字。现在<>将这个新的文件作为默认的输出,因此任何内容将输出到那个文件中◆。while 循环会从旧的文件中读入一行,更新它,再把它输出到新文件中。在一台普通的机器上运行这个程序,几秒钟就能更新上 百个文件。非常强大,不是吗?
2009/10/18 Beckheng Lam <[email protected]> > Cool~~~ ;-) > > bio_gzd wrote: > > OH YEAH.总算是搞定了.贴上代码. > > chop $new_tid=<STDIN>; > > my @FILE = undef; > > open FH, $_tid_file or die "文件打开错误,请检查!!"; > > @FILE = <FH>; > > close (FH); > > open WRTIDFILE,'>', $_tid_file or die "文件打开错误,请检查!!"; > > foreach (@FILE) { > > s/^(TerminalID)\=(\d{8})/$1=$new_tid/g ; > > print WRTIDFILE ; > > } > > close (WRTIDFILE); > > } > > > > 用foreach来遍历一遍整个数组,之后直接写入句柄,关闭..就这样简单.. > > > > On 10月17日, 下午9时00分, bio_gzd <[email protected]> wrote: > > > >> 分开写..试了一下,不行.难道我写错了?有很大可能啊.. > >> my @FILE = open FH, $_tid_file or die "文件打开错误,请检查!!"; > >> close (FH); > >> open WRTIDFILE,'>', $_tid_file or die "文件打开错误,请检查!!"; > >> while (@FILE) { > >> s/^(TerminalID)\=(\d{8})/$1=$new_tid/g ; > >> print WRTIDFILE ; > >> } > >> close (WRTIDFILE); > >> > >> On 10月17日, 上午11时27分, bio_gzd <[email protected]> wrote: > >> > >> > >>> 分开写?试过,但是之前的读没关闭,少了这一步.一会试试.. > >>> > >>> On 10月17日, 上午8时32分, silent <[email protected]> wrote: > >>> > >>>> 额... 你是想改写文件的内容, 那么一边读一边写一个文件句柄恐怕不可以 > >>>> 用两个吧, > >>>> 1种方法是先全读出来放倒变量里, 再关闭文件, 从新以写打开,在把替换过的内容写进去 > >>>> 或者用林时候文件,最后在改名字覆盖 > >>>> > >>>> 2009/10/17 silent <[email protected]>: > >>>> > >>>>> while (<TIDFILE>) { 是表示读的意思 > >>>>> 写入的方法是 print TIDFILE $xxstring > >>>>> > >>>>> 加油! > >>>>> > >>>>> 2009/10/17 bio_gzd <[email protected]>: > >>>>> > >>>>>> 有以下程序段,各位看看有错误没?是一个用正则改写文件内容的 > >>>>>> > >>>>>> unless ($new_tid =~ m/^\d{8,8}[^\w]/) { > >>>>>> print "输入错误或为空行"; > >>>>>> } else { > >>>>>> chop $new_tid; > >>>>>> open TIDFILE, ">$_tid_file" or die "文件打开错误,请检查!!"; > >>>>>> while (<TIDFILE>) { > >>>>>> s/^(TID)\=(\d{8})/$1=$new_tid/g ; > >>>>>> print ; > >>>>>> close (TIDFILE); > >>>>>> } > >>>>>> > >>>>>> > 看看有错没?总是无法写入,如果TIDFILE里面不设置重定向符,直接print输出结果证明替换是正确的,但是一旦要写入到TIDFILE里面要么 > >>>>>> 出错,要么就是0字节.问题出在哪里?麻烦各位指点一下了. > >>>>>> > > > -- > Perl乐事 -- http://www.perlersh.org > 我的博客 -- http://www.perlersh.org/blog.html > > > > > --~--~---------~--~----~------------~-------~--~----~ 您收到此信息是由于您订阅了 Google 论坛“PerlChina Mongers 讨论组”论坛。 要在此论坛发帖,请发电子邮件到 [email protected] 要退订此论坛,请发邮件至 [email protected] 更多选项,请通过 http://groups.google.com/group/perlchina?hl=zh-CN 访问该论坛 -~----------~----~----~----~------~----~------~--~---
