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

📄 ch46_07.htm

📁 the unix power tools
💻 HTM
字号:
<HTML><!--Distributed by F --><HEAD><TITLE>[Chapter 46] 46.7 Quoting and Command-Line Parameters </TITLE><METANAME="DC.title"CONTENT="UNIX Power Tools"><METANAME="DC.creator"CONTENT="Jerry Peek, Tim O'Reilly &amp; Mike Loukides"><METANAME="DC.publisher"CONTENT="O'Reilly &amp; Associates, Inc."><METANAME="DC.date"CONTENT="1998-08-04T21:55:41Z"><METANAME="DC.type"CONTENT="Text.Monograph"><METANAME="DC.format"CONTENT="text/html"SCHEME="MIME"><METANAME="DC.source"CONTENT="1-56592-260-3"SCHEME="ISBN"><METANAME="DC.language"CONTENT="en-US"><METANAME="generator"CONTENT="Jade 1.1/O'Reilly DocBook 3.0 to HTML 4.0"><LINKREV="made"HREF="mailto:online-books@oreilly.com"TITLE="Online Books Comments"><LINKREL="up"HREF="ch46_01.htm"TITLE="46. Shell Script Debugging and Gotchas"><LINKREL="prev"HREF="ch46_06.htm"TITLE="46.6 Watch Out for Bourne Shell -e Bug "><LINKREL="next"HREF="ch46_08.htm"TITLE="46.8 Test Built-In Commands for Failure "></HEAD><BODYBGCOLOR="#FFFFFF"TEXT="#000000"><DIVCLASS="htmlnav"><H1><IMGSRC="gifs/smbanner.gif"ALT="UNIX Power Tools"USEMAP="#srchmap"BORDER="0"></H1><MAPNAME="srchmap"><AREASHAPE="RECT"COORDS="0,0,466,58"HREF="index.htm"ALT="UNIX Power Tools"><AREASHAPE="RECT"COORDS="467,0,514,18"HREF="jobjects/fsearch.htm"ALT="Search this book"></MAP><TABLEWIDTH="515"BORDER="0"CELLSPACING="0"CELLPADDING="0"><TR><TDALIGN="LEFT"VALIGN="TOP"WIDTH="172"><ACLASS="SECT1"HREF="ch46_06.htm"TITLE="46.6 Watch Out for Bourne Shell -e Bug "><IMGSRC="gifs/txtpreva.gif"SRC="gifs/txtpreva.gif"ALT="Previous: 46.6 Watch Out for Bourne Shell -e Bug "BORDER="0"></A></TD><TDALIGN="CENTER"VALIGN="TOP"WIDTH="171"><B><FONTFACE="ARIEL,HELVETICA,HELV,SANSERIF"SIZE="-1">Chapter 46<BR>Shell Script Debugging and Gotchas</FONT></B></TD><TDALIGN="RIGHT"VALIGN="TOP"WIDTH="172"><ACLASS="SECT1"HREF="ch46_08.htm"TITLE="46.8 Test Built-In Commands for Failure "><IMGSRC="gifs/txtnexta.gif"SRC="gifs/txtnexta.gif"ALT="Next: 46.8 Test Built-In Commands for Failure "BORDER="0"></A></TD></TR></TABLE>&nbsp;<HRALIGN="LEFT"WIDTH="515"TITLE="footer"></DIV><DIVCLASS="SECT1"><H2CLASS="sect1"><ACLASS="title"NAME="UPT-ART-0386">46.7 Quoting and Command-Line Parameters </A></H2><DIVCLASS="msgset"><DIVCLASS="msgentry"><DIVCLASS="msg"><DIVCLASS="msgtext"><PCLASS="para"><BCLASS="msgentry.role">Q:</B> <ACLASS="indexterm"NAME="UPT-ART-386-IX-COMMAND-LINE-ARGUMENTS-QUOTING"></A><ACLASS="indexterm"NAME="UPT-ART-386-IX-QUOTING-COMMAND-LINE-PARAMETERS"></A> I need to pass a shell script some arguments with multiple words.I thought that putting<SPANCLASS="link">quotes (<ACLASS="linkend"HREF="ch08_14.htm"TITLE="Bourne Shell Quoting ">8.14</A>)</SPAN>around command-line arguments wouldgroup them.The shell script seems to ignore the quoting, somehow.Here's a simple example:<ACLASS="indexterm"NAME="AUTOID-55708"></A></P><PCLASS="para"><BCLASS="msgentry.role">Q:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">$ <CODECLASS="userinput"><B>cat script</B></CODE>   ...for arg in $*do    echo &quot;Argument is $arg&quot;done$ <CODECLASS="userinput"><B>script '1 2 3' 4</B></CODE>   ...Argument is 1Argument is 2Argument is 3Argument is 4</PRE></BLOCKQUOTE></P></DIV></DIV></DIV><DIVCLASS="msgentry"><DIVCLASS="msg"><DIVCLASS="msgtext"><PCLASS="para"><BCLASS="msgentry.role">A:</B> <ACLASS="indexterm"NAME="AUTOID-55720"></A> This is the way <CODECLASS="literal">$*</CODE> is defined to work. <CODECLASS="literal">$*</CODE> expands to:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">$1 $2</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> [not <CODECLASS="literal">&lt;&quot;&gt;$1&lt;&quot;&gt; &lt;&quot;&gt;$2&lt;&quot;&gt;</CODE><EMCLASS="emphasis">-JP</EM>&nbsp;] if there are two arguments. Hence the <EMCLASS="emphasis">for</EM> loop reads:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">for arg in 1 2 3 4</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> Note that the quotes are gone. What you wanted the shellto see was:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">for arg in '1 2 3' 4</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> You cannot get that, but you can get something that is Good Enough:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <TABLECLASS="screen.co"BORDER="1"><TR><THVALIGN="TOP"><PRECLASS="calloutlist"><ACLASS="co"HREF="ch44_15.htm"TITLE="44.15 Handling Command-Line Arguments in Shell Scripts ">&quot;$@&quot;</A> </PRE></TH><TDVALIGN="TOP"><PRECLASS="screen">for arg in &quot;$@&quot;</PRE></TD></TR></TABLE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> In effect, <CODECLASS="literal">$@</CODE> expands to:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">$1&quot; &quot;$2</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> Putting <CODECLASS="literal">&quot;&quot;</CODE>s around <CODECLASS="literal">$@</CODE>, the effect is:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">for arg in &quot;$1&quot; &quot;$2&quot;</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <ACLASS="indexterm"NAME="AUTOID-55750"></A><ACLASS="indexterm"NAME="AUTOID-55753"></A><ACLASS="indexterm"NAME="AUTOID-55756"></A><ACLASS="indexterm"NAME="AUTOID-55759"></A><ACLASS="indexterm"NAME="AUTOID-55762"></A><ACLASS="indexterm"NAME="AUTOID-55764"></A><ACLASS="indexterm"NAME="AUTOID-55767"></A>Shell quoting is unnecessarily complex. The C shell actually hasthe right idea (variables can be set to<SPANCLASS="link">&quot;word lists&quot; (<ACLASS="linkend"HREF="ch47_05.htm"TITLE="Using C Shell Arrays ">47.5</A>)</SPAN>;<EMCLASS="emphasis">argv</EM> is sucha list), but its defaults and syntax for suppressing them make foran artless programming language:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen"><ACLASS="indexterm"NAME="AUTOID-55774"></A><ACLASS="indexterm"NAME="AUTOID-55777"></A>foreach arg ($argv:q)      # colon q ?!?</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> For the special case of iterating a shell variable over the argumentlist as it stands at the beginning of the iteration, the Bourne shellprovides the construct <CODECLASS="literal">for arg do</CODE>[i.e., no <CODECLASS="literal">in </CODE><CODECLASS="replaceable"><I>list</I></CODE><EMCLASS="emphasis">-JP</EM>&nbsp;]:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">for argdo echo &quot;Argument is $arg&quot;done</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> produces:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">Argument is 1 2 3Argument is 4</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <ACLASS="indexterm"NAME="AUTOID-55791"></A><ACLASS="indexterm"NAME="AUTOID-55794"></A><ACLASS="indexterm"NAME="AUTOID-55797"></A><CODECLASS="literal">&quot;$@&quot;</CODE> is still needed for passing argument lists to other programs.Unfortunately, since <CODECLASS="literal">$@</CODE> is defined as expanding to:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">$1&quot; &quot;$2...$<CODECLASS="replaceable"><I>n-1</I></CODE>&quot; &quot;$<CODECLASS="replaceable"><I>n</I></CODE></PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> (where <CODECLASS="replaceable"><I>n</I></CODE> is the number of arguments), when there are no arguments:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">&quot;$@&quot;</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> expands to:</P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">&quot;&quot;</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> and <CODECLASS="literal">&quot;&quot;</CODE> produces a single argument. [Many UNIX vendorsconsidered this a bug and changed it so that it produces <EMCLASS="emphasis">no</EM>arguments. <EMCLASS="emphasis">-JP</EM>&nbsp;]The best solution for this is to use, for example: </P><PCLASS="para"><BCLASS="msgentry.role">A:</B> <BLOCKQUOTECLASS="screen"><PRECLASS="screen">% <CODECLASS="userinput"><B>cat bin/okeeffe</B></CODE>#! /bin/shexec rsh okeeffe.berkeley.edu -l torek ${1+&quot;$@&quot;}%</PRE></BLOCKQUOTE></P><PCLASS="para"><BCLASS="msgentry.role">A:</B> The construct <CODECLASS="literal">${1+&quot;$@&quot;}</CODE> means &quot;expand <CODECLASS="literal">$1</CODE>, but if <CODECLASS="literal">$1</CODE>is defined, use <CODECLASS="literal">&quot;$@&quot;</CODE> instead.&quot; [You don't need this onBourne shells with the &quot;bug fix&quot; I mentioned. <EMCLASS="emphasis">-JP</EM>&nbsp;]Hence, if there are no arguments, we get <CODECLASS="literal">$1</CODE> (which isnothing and produces no arguments), otherwise we get <CODECLASS="literal">&quot;$@&quot;</CODE> (whichexpands as above). <CODECLASS="literal">${</CODE><CODECLASS="replaceable"><I>var</I></CODE><CODECLASS="literal">+</CODE><CODECLASS="replaceable"><I>instead</I></CODE><CODECLASS="literal">}</CODE>is one of several <EMCLASS="emphasis">sh</EM><SPANCLASS="link">\*(lqexpansion shortcuts\*(rq (<ACLASS="linkend"HREF="ch45_12.htm"TITLE="Parameter Substitution ">45.12</A>)</SPAN>.Another more generally useful one is<CODECLASS="literal">${</CODE><CODECLASS="replaceable"><I>var-default</I></CODE><CODECLASS="literal">}</CODE>,which expands to <CODECLASS="literal">$</CODE><CODECLASS="replaceable"><I>var</I></CODE>, but if <EMCLASS="emphasis">var</EM> is not set, to<CODECLASS="replaceable"><I>default</I></CODE> instead.All of these can be found in the manual for <EMCLASS="emphasis">sh</EM>, which is worthreading several times, experimenting as you go.</P></DIV></DIV></DIV></DIV><ACLASS="indexterm"NAME="AUTOID-55843"></A><ACLASS="indexterm"NAME="AUTOID-55844"></A><DIVCLASS="sect1info"><PCLASS="SECT1INFO">- <SPANCLASS="authorinitials">CT</SPAN> <SPANCLASS="bibliomisc">in <ACLASS="systemitem.newsgroup"HREF="news:comp.unix.questions">comp.unix.questions</A> on Usenet, 18 March 1988</SPAN></P></DIV></DIV><DIVCLASS="htmlnav"><P></P><HRALIGN="LEFT"WIDTH="515"TITLE="footer"><TABLEWIDTH="515"BORDER="0"CELLSPACING="0"CELLPADDING="0"><TR><TDALIGN="LEFT"VALIGN="TOP"WIDTH="172"><ACLASS="SECT1"HREF="ch46_06.htm"TITLE="46.6 Watch Out for Bourne Shell -e Bug "><IMGSRC="gifs/txtpreva.gif"SRC="gifs/txtpreva.gif"ALT="Previous: 46.6 Watch Out for Bourne Shell -e Bug "BORDER="0"></A></TD><TDALIGN="CENTER"VALIGN="TOP"WIDTH="171"><ACLASS="book"HREF="index.htm"TITLE="UNIX Power Tools"><IMGSRC="gifs/txthome.gif"SRC="gifs/txthome.gif"ALT="UNIX Power Tools"BORDER="0"></A></TD><TDALIGN="RIGHT"VALIGN="TOP"WIDTH="172"><ACLASS="SECT1"HREF="ch46_08.htm"TITLE="46.8 Test Built-In Commands for Failure "><IMGSRC="gifs/txtnexta.gif"SRC="gifs/txtnexta.gif"ALT="Next: 46.8 Test Built-In Commands for Failure "BORDER="0"></A></TD></TR><TR><TDALIGN="LEFT"VALIGN="TOP"WIDTH="172">46.6 Watch Out for Bourne Shell -e Bug </TD><TDALIGN="CENTER"VALIGN="TOP"WIDTH="171"><ACLASS="index"HREF="index/idx_0.htm"TITLE="Book Index"><IMGSRC="gifs/index.gif"SRC="gifs/index.gif"ALT="Book Index"BORDER="0"></A></TD><TDALIGN="RIGHT"VALIGN="TOP"WIDTH="172">46.8 Test Built-In Commands for Failure </TD></TR></TABLE><HRALIGN="LEFT"WIDTH="515"TITLE="footer"><IMGSRC="gifs/smnavbar.gif"SRC="gifs/smnavbar.gif"USEMAP="#map"BORDER="0"ALT="The UNIX CD Bookshelf Navigation"><MAPNAME="map"><AREASHAPE="RECT"COORDS="0,0,73,21"HREF="../index.htm"ALT="The UNIX CD Bookshelf"><AREASHAPE="RECT"COORDS="74,0,163,21"HREF="index.htm"ALT="UNIX Power Tools"><AREASHAPE="RECT"COORDS="164,0,257,21"HREF="../unixnut/index.htm"ALT="UNIX in a Nutshell"><AREASHAPE="RECT"COORDS="258,0,321,21"HREF="../vi/index.htm"ALT="Learning the vi Editor"><AREASHAPE="RECT"COORDS="322,0,378,21"HREF="../sedawk/index.htm"ALT="sed &amp; awk"><AREASHAPE="RECT"COORDS="379,0,438,21"HREF="../ksh/index.htm"ALT="Learning the Korn Shell"><AREASHAPE="RECT"COORDS="439,0,514,21"HREF="../lrnunix/index.htm"ALT="Learning the UNIX Operating System"></MAP></DIV></BODY></HTML>

⌨️ 快捷键说明

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