函数名 fork 调用语法 procid = fork(); 解说 创建程序的两个拷贝--父进程和子进程--同时运行。子进程返回零,父进程返回非零值,此值为子程序的进程ID号。 例子 $retval = fork(); if ($retval == 0) { # this is the child process exit; # this terminates the child process } else { # this is the parent process } 结果输出 无
函数名 pipe 调用语法 pipe (infile, outfile); 解说 与fork合用,给父进程和子进程提供通信的方式。送到outfile文件变量的信息可以通过infile文件变量读取。步骤: 1、调用pipe 2、用fork将程序分成父进程和子进程 3、一个进程关掉infile,另一个关掉outfile 例子 pipe (INPUT, OUTPUT); $retval = fork(); if ($retval != 0) { # this is the parent process close (INPUT); print ("Enter a line of input:\n"); $line = ; print OUTPUT ($line); } else { # this is the child process close (OUTPUT); $line = ; print ($line); exit (0); } 结果输出 $ program Enter a line of input: Here is a test line Here is a test line $
函数名 die 调用语法 die (message); 解说 终止程序并向STDERR输出错误信息。message可以为字符串或列表。如果最后一个参数不包含换行符,则程序文件名和行号也被输出。 例子 die ("Cannot open input file"); 结果输出 Cannot open input file at myprog line 6.
函数名 waitpid 调用语法 waitpid (procid, waitflag); 解说 暂停程序执行,等待特定的子进程终止。procid为等待的进程ID 例子 $procid = fork(); if ($procid == 0) { # this is the child process print ("this line is printed first\n"); exit(0); } else { # this is the parent process waitpid ($procid, 0); print ("this line is printed last\n"); } 结果输出 $ program this line is printed first this line is printed last $ 4、其它控制函数
函数名 pack 调用语法 formatstr = pack(packformat, list); 解说 把一个列表或数组以在实际机器存贮格式或C等编程语言使用的格式转化(包装)到一个简单变量中。参数packformat包含一个或多个格式字符,列表中每个元素对应一个,各格式字符间可用空格或tab隔开,因为pack忽略空格。 除了格式a、A和@外,重复使用一种格式多次可在其后加个整数,如: $twoints = pack ("i2", 103, 241); 把同一格式应用于所有的元素则加个*号,如: $manyints = pack ("i*", 14, 26, 11, 83); 对于a和A而言,其后的整数表示要创建的字符串长度,重复方法如下: $strings = pack ("a6" x 2, "test1", "test2"); 格式@的情况比较特殊,其后必须加个整数,该数表示字符串必须的长度,如果长度不够,则用空字符(null)补足,如: $output = pack ("a @6 a", "test", "test2"); pack函数最常见的用途是创建可与C程序交互的数据,例如C语言中字符串均以空字符(null)结尾,创建这样的数据可以这样做: $Cstring = pack ("ax", $mystring); 下表是一些格式字符与C中数据类型的等价关系: 字符 等价C数据类型 C char d double f float i int I unsigned int (or unsigned) l long L unsigned long s short S unsigned short 完整的格式字符见下表。
格式字符 描述 a 用空字符(null)补足的字符串 A 用空格补足的字符串 b 位串,低位在前 B 位串,高位在前 c 带符号字符(通常-128~127) C 无符号字符(通常8位) d 双精度浮点数 f 单精度浮点数 h 十六进制数串,低位在前 H 十六进制数串,高位在前 i 带符号整数 I 无符号整数 l 带符号长整数 L 无符号长整数 n 网络序短整数 N 网络序长整数 p 字符串指针 s 带符号短整数 S 无符号短整数 u 转化成uuencode格式 v VAX序短整数 V VAX序长整数 x 一个空字节 X 回退一个字节 @ 以空字节(null)填充
1: #!/usr/local/bin/perl 2: 3: open (STDOUT, ">file1") || die ("open STDOUT failed"); 4: open (STDERR, ">&STDOUT") || die ("open STDERR failed"); 5: print STDOUT ("line 1\n"); 6: print STDERR ("line 2\n"); 7: close (STDOUT); 8: close (STDERR);
运行后,文件file1中的内容为: line 2 line 1 可以看到,这两行并未按我们想象的顺序存贮,为什么呢?我们来分析一下这段程序。 第3行重定向标准输出文件,方法是打开文件file1将它与文件变量STDOUT关联,这也关闭了标准输出文件。第4行重定向标准错误文件,参数>&STDOUT告诉Perl解释器使用已打开并与STDOUT关联的文件,即文件变量STDERR指向与STDOUT相同的文件。第5、6行分别向STDOUT和STDERR写入数据,因为这两个文件变量指向同一个文件,故两行字符串均写到文件file1中,但顺序却是错误的,怎么回事呢? 问题在于UNIX对输出的处理上。当使用print(或其它函数)写入STDOUT等文件时,UNIX操作系统真正所做的是把数据拷贝到一片特殊的内存即缓冲区中,接下来的输出操作继续写入缓冲区直到写满,当缓冲区满了,就把全部数据实际输出。象这样先写入缓冲区再把整个缓冲区的内容输出比每次都实际输出所花费的时间要少得多,因为一般来说,I/O比内存操作慢得多。 程序结束时,任何非空的缓冲区都被输出,然而,系统为STDOUT和STDERR分别维护一片缓冲区,并且先输出STDERR的内容,因此存贮在STDERR的缓冲区中的内容line 2出现在存贮在STDOUT的缓冲区中的内容line 1之前。 为了解决这个问题,可以告诉Perl解释器不对文件使用缓冲,方法为: 1、用select函数选择文件 2、把值1赋给系统变量$| 系统变量$|指定文件是否进行缓冲而不管其是否应该使用缓冲。如果$|为非零值则不使用缓冲。$|与系统变量$~和$^协同工作,当未调用select函数时,$|影响当前缺省文件。下例保证了输出的次序:
1 : #!/usr/local/bin/perl 2 : 3 : open (STDOUT, ">file1") || die ("open STDOUT failed"); 4 : open (STDERR, ">&STDOUT") || die ("open STDERR failed"); 5 : $| = 1; 6 : select (STDERR); 7 : $| = 1; 8 : print STDOUT ("line 1\n"); 9 : print STDERR ("line 2\n"); 10: close (STDOUT); 11: close (STDERR);
程序运行后,文件file1中内容为: line 1 line 2 第5行将$|赋成1,告诉Perl解释器当前缺省文件不进行缓冲,因为未调用select,当前的缺省文件为重定向到文件file1的STDOUT。第6行将当前缺省文件设为STDERR,第7行又设置$|为1,关掉了重定向到file1的标准错误文件的缓冲。由于STDOUT和STDERR的缓冲均被关掉,向其的输出立刻被写到文件中,因此line 1出现在第一行。 指定读写权限 打开一个既可读又可写的文件方法是在文件名前加上"+>",如下: open (READWRITE, "+>file1"); 此语句打开既可读又可写的文件file1,即可以重写其中的内容。文件读写操作最好与库函数seek和tell一起使用,这样可以跳到文件任何一点。 注:也可用前缀"+<"指定可读写权限。 close函数 用于关闭打开的文件。当用close关闭管道,即重定向的命令时,程序等待重定向的命令结束,如: open (MYPIPE, "cat file*|"); close (MYPIPE); 当关闭此文件变量时,程序暂停运行,直到命令cat file*运行完毕。 print, printf和write函数 print是这三个函数中最简单的,它向指定的文件输出,如果未指定,则输出到当前缺省文件中,如: print ("Hello, there!\n"); print OUTFILE ("Hello, there!\n"); 第一句输出到当前缺省文件中,若未调用select,则为STDOUT。第二句输出到由文件变量OUTFILE指定的文件中。 printf函数先格式化字符串再输出到指定文件或当前缺省文件中,如: printf OUTFILE (“You owe me %8.2f", $owing); 此语句取出变量$owing的值并替换掉串中的%8.2f,%8.2f是域格式的例子,把$owing的值看作浮点数。 write函数使用输出格式把信息输出到文件中,如: select (OUTFILE); $~ = "MYFORMAT"; write; 关于printf和write,详见《第x章 格式化输出》。 select函数 select函数将通过参数传递的文件变量指定为新的当前缺省文件,如: select (MYFILE); 这样,MYFILE就成了当前缺省文件,当对print、write和printf的调用未指定文件时,就输出到MYFILE中。 eof函数 eof函数查看最后一次读文件操作是否为文件最后一个记录,如果是,则返回非零值,如果文件还有内容,返回零。 一般情况下,对eof的调用不加括号,因为eof和eof()是等效的,但与<>操作符一起使用时,eof和eof()就不同了。现在我们来创建两个文件,分别叫做file1和file2。file1的内容为: This is a line from the first file. Here is the last line of the first file. file2的内容为: This is a line from the second and last file. Here is the last line of the last file. 下面就来看一下eof和eof()的区别,第一个程序为:
1: #!/usr/local/bin/perl 2: 3: while ($line = <>) { 4: print ($line); 5: if (eof) { 6: print ("-- end of current file --\n"); 7: } 8: }
运行结果如下:
$ program file1 file2 This is a line from the first file. Here is the last line of the first file. -- end of current file -- This is a line from the second and last file. Here is the last line of the last file. -- end of current file -- $
下面把eof改为eof(),第二个程序为:
1: #!/usr/local/bin/perl 2: 3: while ($line = <>) { 4: print ($line); 5: if (eof()) { 6: print ("-- end of output --\n"); 7: } 8: }
运行结果如下:
$ program file1 file2 This is a line from the first file. Here is the last line of the first file. This is a line from the second and last file. Here is the last line of the last file. -- end of output --$