⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 x13369.html

📁 BASH Shell 编程 经典教程 《高级SHELL脚本编程》中文版
💻 HTML
字号:
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"><HTML><HEAD><TITLE>使用exec</TITLE><METANAME="GENERATOR"CONTENT="Modular DocBook HTML Stylesheet Version 1.79"><LINKREL="HOME"TITLE="高级Bash脚本编程指南"HREF="index.html"><LINKREL="UP"TITLE="I/O重定向"HREF="io-redirection.html"><LINKREL="PREVIOUS"TITLE="I/O重定向"HREF="io-redirection.html"><LINKREL="NEXT"TITLE="代码块重定向"HREF="redircb.html"></HEAD><BODYCLASS="SECT1"BGCOLOR="#FFFFFF"TEXT="#000000"LINK="#0000FF"VLINK="#840084"ALINK="#0000FF"><DIVCLASS="NAVHEADER"><TABLESUMMARY="Header navigation table"WIDTH="100%"BORDER="0"CELLPADDING="0"CELLSPACING="0"><TR><THCOLSPAN="3"ALIGN="center">高级Bash脚本编程指南: 一本深入学习shell脚本艺术的书籍</TH></TR><TR><TDWIDTH="10%"ALIGN="left"VALIGN="bottom"><AHREF="io-redirection.html"ACCESSKEY="P">前一页</A></TD><TDWIDTH="80%"ALIGN="center"VALIGN="bottom">16. I/O重定向</TD><TDWIDTH="10%"ALIGN="right"VALIGN="bottom"><AHREF="redircb.html"ACCESSKEY="N">下一页</A></TD></TR></TABLE><HRALIGN="LEFT"WIDTH="100%"></DIV><DIVCLASS="SECT1"><H1CLASS="SECT1"><ANAME="AEN13369">16.1. 使用<BCLASS="COMMAND">exec</B></A></H1><P><ANAME="USINGEXECREF"></A></P><P><BCLASS="COMMAND">exec &#60;filename</B>命令会将<TTCLASS="FILENAME">stdin</TT>重定向到文件中. 			从这句开始, 所有的<TTCLASS="FILENAME">stdin</TT>就都来自于这个文件了, 		而不是标准输入(通常都是键盘输入). 		这样就提供了一种按行读取文件的方法, 		并且可以使用<AHREF="sedawk.html#SEDREF">sed</A>和/或<AHREF="awk.html#AWKREF">awk</A>来对每一行进行分析. </P><DIVCLASS="EXAMPLE"><HR><ANAME="REDIR1"></A><P><B>例子 16-1. 使用<BCLASS="COMMAND">exec</B>重定向<TTCLASS="FILENAME">stdin</TT></B></P><TABLEBORDER="0"BGCOLOR="#E0E0E0"WIDTH="100%"><TR><TD><FONTCOLOR="#000000"><PRECLASS="PROGRAMLISTING">#!/bin/bash# 使用'exec'重定向stdin. exec 6&#60;&#38;0          # 将文件描述符#6与stdin链接起来.                    # 保存stdin. exec &#60; data-file   # stdin被文件"data-file"所代替. read a1            # 读取文件"data-file"的第一行. read a2            # 读取文件"data-file"的第二行. echoecho "Following lines read from file."echo "-------------------------------"echo $a1echo $a2echo; echo; echoexec 0&#60;&#38;6 6&#60;&#38;-#  现在将stdin从fd #6中恢复, 因为刚才我们把stdin重定向到#6了, #+ 然后关闭fd #6 ( 6&#60;&#38;- ), 好让这个描述符继续被其他进程所使用. ## &#60;&#38;6 6&#60;&#38;-    这么做也可以. echo -n "Enter data  "read b1  # 现在"read"已经恢复正常了, 就是能够正常的从stdin中读取.echo "Input read from stdin."echo "----------------------"echo "b1 = $b1"echoexit 0</PRE></FONT></TD></TR></TABLE><HR></DIV><P>同样的, <BCLASS="COMMAND">exec &#62;filename</B>命令将会把<TTCLASS="FILENAME">stdout</TT>重定向到一个指定的文件中. 	  这样所有命令的输出就都会发送到那个指定的文件, 	  而不是<TTCLASS="FILENAME">stdout</TT>. </P><DIVCLASS="IMPORTANT"><P></P><TABLECLASS="IMPORTANT"WIDTH="100%"BORDER="0"><TR><TDWIDTH="25"ALIGN="CENTER"VALIGN="TOP"><IMGSRC="./images/important.gif"HSPACE="5"ALT="Important"></TD><TDALIGN="LEFT"VALIGN="TOP"><P>			<BCLASS="COMMAND">exec N &#62; filename</B>会影响整个脚本或<EM>当前shell</EM>. 			对于这个指定<AHREF="special-chars.html#PROCESSIDREF">PID</A>的脚本或shell来说, 			从这句命令执行之后, 就会重定向到这个文件中, 			然而 . . .        </P><P>	  <BCLASS="COMMAND">N &#62; filename</B>只会影响新fork出来的进程, 而不会影响整个脚本或shell. 	  not the entire script or shell.        </P><P>感谢你, Ahmed Darwish, 指出这个问题. </P></TD></TR></TABLE></DIV><DIVCLASS="EXAMPLE"><HR><ANAME="REASSIGNSTDOUT"></A><P><B>例子 16-2. 使用<BCLASS="COMMAND">exec</B>来重定向<TTCLASS="FILENAME">stdout</TT></B></P><TABLEBORDER="0"BGCOLOR="#E0E0E0"WIDTH="100%"><TR><TD><FONTCOLOR="#000000"><PRECLASS="PROGRAMLISTING">#!/bin/bash# reassign-stdout.shLOGFILE=logfile.txtexec 6&#62;&#38;1           # 将fd #6与stdout链接起来.                     # 保存stdout. exec &#62; $LOGFILE     # stdout就被文件"logfile.txt"所代替了. # ----------------------------------------------------------- ## 在这块中所有命令的输出都会发送到文件$LOGFILE中. echo -n "Logfile: "dateecho "-------------------------------------"echoecho "Output of \"ls -al\" command"echols -alecho; echoecho "Output of \"df\" command"echodf# ----------------------------------------------------------- #exec 1&#62;&#38;6 6&#62;&#38;-      # 恢复stdout, 然后关闭文件描述符#6. echoecho "== stdout now restored to default == "echols -alechoexit 0</PRE></FONT></TD></TR></TABLE><HR></DIV><DIVCLASS="EXAMPLE"><HR><ANAME="UPPERCONV"></A><P><B>例子 16-3. 使用<BCLASS="COMMAND">exec</B>在同一个脚本中重定向<TTCLASS="FILENAME">stdin</TT>和<TTCLASS="FILENAME">stdout</TT></B></P><TABLEBORDER="0"BGCOLOR="#E0E0E0"WIDTH="100%"><TR><TD><FONTCOLOR="#000000"><PRECLASS="PROGRAMLISTING">#!/bin/bash# upperconv.sh# 将一个指定的输入文件转换为大写. E_FILE_ACCESS=70E_WRONG_ARGS=71if [ ! -r "$1" ]     # 判断指定的输入文件是否可读? then  echo "Can't read from input file!"  echo "Usage: $0 input-file output-file"  exit $E_FILE_ACCESSfi                   #  即使输入文件($1)没被指定                     #+ 也还是会以相同的错误退出(为什么?). if [ -z "$2" ]then  echo "Need to specify output file."  echo "Usage: $0 input-file output-file"  exit $E_WRONG_ARGSfiexec 4&#60;&#38;0exec &#60; $1            # 将会从输入文件中读取. exec 7&#62;&#38;1exec &#62; $2            # 将写到输出文件中.                      # 假设输出文件是可写的(添加检查?). # -----------------------------------------------    cat - | tr a-z A-Z   # 转换为大写. #   ^^^^^                # 从stdin中读取. #           ^^^^^^^^^^   # 写到stdout上. # 然而, stdin和stdout都被重定向了. # -----------------------------------------------exec 1&#62;&#38;7 7&#62;&#38;-       # 恢复stout.exec 0&#60;&#38;4 4&#60;&#38;-       # 恢复stdin.# 恢复之后, 下边这行代码将会如预期的一样打印到stdout上. echo "File \"$1\" written to \"$2\" as uppercase conversion."exit 0</PRE></FONT></TD></TR></TABLE><HR></DIV><P>I/O重定向是一种避免可怕的<AHREF="subshells.html#PARVIS">子shell中不可访问变量</A>问题的方法.       </P><DIVCLASS="EXAMPLE"><HR><ANAME="AVOIDSUBSHELL"></A><P><B>例子 16-4. 避免子shell</B></P><TABLEBORDER="0"BGCOLOR="#E0E0E0"WIDTH="100%"><TR><TD><FONTCOLOR="#000000"><PRECLASS="PROGRAMLISTING">#!/bin/bash# avoid-subshell.sh# 由Matthew Walker所提出的建议. Lines=0echocat myfile.txt | while read line;  #  (译者注: 管道会产生子shell)                 do {                   echo $line                   (( Lines++ ));  #  增加这个变量的值                                   #+ 但是外部循环却不能访问.                                    #  子shell问题.                  }                 doneecho "Number of lines read = $Lines"     # 0                                         # 错误!echo "------------------------"exec 3&#60;&#62; myfile.txtwhile read line &#60;&#38;3do {  echo "$line"  (( Lines++ ));                   #  增加这个变量的值                                   #+ 现在外部循环就可以访问了.                                    #  没有子shell, 现在就没问题了. }doneexec 3&#62;&#38;-echo "Number of lines read = $Lines"     # 8echoexit 0# 下边这些行是这个脚本的结果, 脚本是不会走到这里的. $ cat myfile.txtLine 1.Line 2.Line 3.Line 4.Line 5.Line 6.Line 7.Line 8.</PRE></FONT></TD></TR></TABLE><HR></DIV></DIV><DIVCLASS="NAVFOOTER"><HRALIGN="LEFT"WIDTH="100%"><TABLESUMMARY="Footer navigation table"WIDTH="100%"BORDER="0"CELLPADDING="0"CELLSPACING="0"><TR><TDWIDTH="33%"ALIGN="left"VALIGN="top"><AHREF="io-redirection.html"ACCESSKEY="P">前一页</A></TD><TDWIDTH="34%"ALIGN="center"VALIGN="top"><AHREF="index.html"ACCESSKEY="H">首页</A></TD><TDWIDTH="33%"ALIGN="right"VALIGN="top"><AHREF="redircb.html"ACCESSKEY="N">下一页</A></TD></TR><TR><TDWIDTH="33%"ALIGN="left"VALIGN="top">I/O重定向</TD><TDWIDTH="34%"ALIGN="center"VALIGN="top"><AHREF="io-redirection.html"ACCESSKEY="U">上一级</A></TD><TDWIDTH="33%"ALIGN="right"VALIGN="top">代码块重定向</TD></TR></TABLE></DIV></BODY></HTML>

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -